C语言实现十六进制与二进制的相互转换

目录
  • 十六进制->二进制
  • 二进制->十六进制

本文中的代码可以将文件中的十六进制存储与二进制存储相互转换。

十六进制->二进制

原理是:每两位存储为一个字符(char)保存。

因为十六进制数最大为 f,即 15,在内存中只需要 4 位就可以表示。而一般情况下一个字符是占一个字节 8 位,所以正好可以存储十六进制两位。

举个栗子:

在文件中存储十六进制为ab,ab转换为二进制,就是1010 1011,刚好八位,可以联想到ASCII码,用一个字符可以表示。

注意转为二进制后,前32个为不可见字符,附图ASCII码。

(将十六进制转为二进制还会压缩一倍的空间?因为ab本来占两个字节,现在转成二进制,即一个char字符,只占用一个字节大小。但是转换过程中时间消耗,我不太确定是否优化了)

#include <stdio.h>

int main() {

    FILE * in  = fopen("./data.txt", "r");
    FILE * out  = fopen("./data", "w");

    while (1) {
        char c;
        unsigned char d = 0;

        for (int i = 0; i < 2; ++i) {
            // 从 in 读取 1 个大小为 1 字节数据保存在 c
            if (fread(&c, 1, 1, in) == 0) {
                fclose(in);
                fclose(out);
                return 0;
            }
        	// 读到空格或者换行索引需要回退
            if (c == ' ' || c == '\n') {
                i--;
                continue;
            }
            // 将读到的十六进制字符转成具体的十进制数字
            if (c >= '0' && c <= '9') {
                c -= '0';
            } else if (c >= 'a' && c <= 'f') {
                c -= 'a';
                c += 10;
            } else {
                printf("error");
            }
            //printf("c -> %d  \n", c);
            d <<= 4;
            d |= c;
        }
        //printf("-------> %c\n", d);
        fwrite(&d, 1, 1, out);
    }

    return 0;

}

核心代码:

d <<= 4 and d |= c

解释:

内层for循环为两次,c读取两次:

我们假设第一次读到a字符,第二次读到b字符。

(或运算:参加运算的两个对象只要有一个为1,其值为1)

操作
第一次循环开始 : d (0000 0000) c(0000 1010)
d <<= 4 : d (0000 0000)     c(0000 1010)
d |= c : d (0000 1010) c(0000 1010)
第二次循环开始: d (0000 1010) c(0000 1011)
d <<= 4 :  d (1010 0000)     c(0000 1011)
d |= c :  d (1010 1011) c(0000 1011)
两次循环结束 将d写入文件,转换完成。  

如果想知道转换是否正确的话,可以用 hexdump -C + 文件名(data) 检查一下是否正确。(linux命令)

二进制->十六进制

这个就是上边操作相反的过程,读取二进制文件,然后转成十六进制字符保存。

#include <stdio.h>

int main() {

    FILE * in  = fopen("./data.txt", "r");
    FILE * out  = fopen("./data", "w");

    char space = ' ';
    char enter = '\n';

    int idx = 0;
    while (1) {
        char c[2];
        unsigned char d = 0;

        for (int i = 0; i < 4; ++i) {
            // 读取一个字节
            if (fread(&d, 1, 1, in) == 0) {
                fclose(in);
                fclose(out);
                return 0;
            }

            // 一个二进制字节转回两个十六进制字符
            char mask = 0xf;
            c[0] = d >> 4;      // 将低位移走就是该字节保存的第一个十六进制字符
            c[1] = d & mask;    // 保留 d 的低位就是该字节保存的第二个十六进制字符

            // 将十进制数字转回对应的十六进制字符(与上一步转换的代码相反)
            if (c[0] >= 0 && c[0] <= 9) {
                c[0] += '0';
            } else {
                c[0] -= 10;
                c[0] += 'a';
            }
            if (c[1] >= 0 && c[1] <= 9) {
                c[1] += '0';
            } else {
                c[1] -= 10;
                c[1] += 'a';
            }

            // 从 c 地址开始读取两个字节,写到 out
            fwrite(&c, 1, 2, out);
        }
        // 写空格或者换行 保持格式
        idx++;
        if (idx == 4){
            idx = 0;
            fwrite(&enter, 1, 1, out);
        } else {
            fwrite(&space, 1, 1, out);
        }
    }
    return 0;

}

mask的二进制是0000 1111。

到此这篇关于C语言实现十六进制与二进制的相互转换的文章就介绍到这了,更多相关C语言 十六进制二进制互转内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • c语言中十进制转二进制显示小工具的实现代码

    计算器在显示二进制位数时候,如果开头是0.是不会显示的,对于在单片机混的人,这个有时候很麻烦,所以写个小工具. 功能就是输入十进制数字,然后显示出2进制,每显示4位一次空格,可以调整位数范围(8的倍数) 如果有谁能知道linux下类似win7的那个计算器,麻烦回复告知一下吧.很是感谢~ 例如: $ dec2bin 1 135Num 8 Binary is : 0B1000 0111done!=============================== $ dec2bin 2 135Num 16

  • c语言中十六进制转二进制显示的实现方法

    复制代码 代码如下: //====================================== //输出格式: hex2bin 5e. //得到: 0101 1110 //====================================== #include <stdio.h>#include <limits.h> char *bitstr(char *, void const *, size_t); int main(int argc, char **argv){

  • C语言中十六进制转十进制两种实现方法

    C语言 · 十六进制转十进制 问题描述 从键盘输入一个不超过8位的正的十六进制数字符串,将它转换为正的十进制数后输出. 注:十六进制数中的10~15分别用大写的英文字母A.B.C.D.E.F表示. 样例输入 FFFF 样例输出 65535 思路:感觉自己的下面两个方法都对,但是···不说了[狡诈]... 方案一: #include<stdio.h> #include<math.h> #include<string.h> int main(){ char s[50]; s

  • C语言十进制转二进制代码实例

    用C语言实现将十进制转化为二进制,并统计转换后的二进制码中1的个数. #include <stdio.h> int binaryNum[16]; //存放转换后得到的二进制码 int count=0; //计数十进制整数被2除的次数 int oneCount=0; //得到的二进制码中1的个数 void main(){ int num; printf("输入一个十进制的整数:"); scanf("%d",&num); while( (num/2)

  • C语言实现十六进制与二进制的相互转换

    目录 十六进制->二进制 二进制->十六进制 本文中的代码可以将文件中的十六进制存储与二进制存储相互转换. 十六进制->二进制 原理是:每两位存储为一个字符(char)保存. 因为十六进制数最大为 f,即 15,在内存中只需要 4 位就可以表示.而一般情况下一个字符是占一个字节 8 位,所以正好可以存储十六进制两位. 举个栗子: 在文件中存储十六进制为ab,ab转换为二进制,就是1010 1011,刚好八位,可以联想到ASCII码,用一个字符可以表示. 注意转为二进制后,前32个为不可见

  • javascript十六进制及二进制转化的方法

    本文实例讲述了javascript十六进制及二进制转化的方法.分享给大家供大家参考.具体实现方法如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"

  • C语言实现十六进制转换为十进制的方法详解

    目录 题目描述 补充知识 优化算法 补充办法 题目描述 输入一个十六进制数字串,将其转换成为对应的整数并输出转换结果,遇到非十六进制数字或字符串结束符('\0')结束转换. 注意: 输入的字符串的长度不会超过100:转换后的值不会超出int类型的范围. 1.测试输入:55 //程序的输入不会有十六进制的前缀 预期输出:85 2.测试输入:f1r2 //程序的输入不会有十六进制的前缀 预期输出:241 补充知识 十六进制转换为十进制的做法: 算法分析 #include<stdio.h> #inc

  • C语言中字符串和数字的相互转换实现代码

    1.数字转换为字符串sprintf 跟printf 在用法上几乎一样,只是打印的目的地不同而已,前者打印到字符串中,后者则直接在命令行上输出.sprintf 是个变参函数,定义如下: int sprintf( char *buffer, const char *format [, argument] ... ); 除了前两个参数类型固定外,后面可以接任意多个参数.printf 和sprintf 都使用格式化字符串来指定串的格式,在格式串内部使用一些以"%"开头的格式说明符(format

  • C# 进制转换的实现(二进制、十六进制、十进制互转)

    由于二进制数在C#中无法直接表示,所以所有二进制数都用一个字符串来表示 例如: 二进制: 1010 表示为 字符串:"1010"  int d = 10; //十进制转二进制字符串 Console.WriteLine(Convert.ToString(d,2)); //输出: 1010 //十进制转十六进制字符串 Console.WriteLine(Convert.ToString(d,16)); //输出: a //二进制字符串转十进制数 string bin = "101

  • 一篇教程教你学会Python进制转换(十进制转二进制、八进制、十六进制)

    一.导言 导语: 在计算机进行数据交换时,常常会有一个进制转换的过程,我们知道计算机只认0 和 1.在内存系统中,基本基于二进制进行运算的,但是有时候数据过于庞大,为了方便存储管理,计算机会使用十六进制存储数据,但是怎么实现数据转换呢? 我们人类由十根手指头,所以自然就使用十进制啦,每当我们数数字到10之后,于是就重0 开始继续数,所以逢十进一就这么来了. 对于其它进制呢,也是同样的,比如最常见的二进制,也就是逢二进一,慢慢来,依次类推,所以数学的世界是多么的奇妙呀! 今天给大家带来的是使用 P

  • 科学知识:二进制、八进制、十进制、十六进制转换

    一. 十进制与二进制之间的转换 (1) 十进制转换为二进制,分为整数部分和小数部分 ① 整数部分 方法:除2取余,逆序排列,即每次将整数部分除以2,余数为该位权上的数,而商继续除以2,余数又为上一个位权上的数,这个步骤一直持续下去,直到商为0为止,最后读数时候,从最后一个余数读起,一直到最前面的一个余数.下面举例: 例:将十进制的168转换为二进制 得出结果 将十进制的168转换为二进制,(10101000) 第一步,将168除以2,商84,余数为0. 第二步,将商84除以2,商42余数为0.

随机推荐