问题引出

C语言中,在需要用到16进制数据的时候,可以通过printf函数的%x格式打印数据的16进制形式。在某些位标记、位操作的场合,需要用到2进制格式的数据,但printf函数不能输出2进制格式,虽然可以通过使用itoa_itoa的方法转为2进制的字符串打印,但显示的长度是不固定的,无法显示有效数位前面的0。

例如:现在需要打印数字258的2进制格式,且需要将32位全部显示出来,即想要得到结果00000000 00000000 00000001 00000010,而使用_itoa的方法和打印结果为:

int a = 258;
char s[32];
_itoa(a, s, 2);
printf("a --> %s\n", s);
a --> 100000010

不是我们想要的格式!

数据存储原理探析

那要怎么办呢?自己写个小程序吧,思路如下:

首先弄清楚数据在计算机中是如何存储的,对应int型数字,在32或64位计算机中都占4个字节,而计算机中的数据存储是以字节(Byte)为单位,1个字节包含8个位(bit),例如,数字258的16进制形式为0x00000102,2进制形式为:00000000 00000000 00000001 00000010,其在计算机内存中的存储方式如图所示:

右侧的16进制数是内存的地址,向上递增,方框里的二进制数是内存单元实际存储的字节内存,我们可以通过程序测试验证一下,因为unsigned char或char类型在系统是占用一个字节,因此可以定义该变量的指针,分别指向int的4个字节,打印其内存地址和实际存储的内存进行验证,代码如下:

int a = 258;
//使用unsigned char来验证int的每一个字节
unsigned char *p1 = (unsigned char*)&a;   //获取a的首地址
unsigned char *p2 = (unsigned char*)&a+1; //获取a的首地址的后一个字节地址
unsigned char *p3 = (unsigned char*)&a+2; //获取a的首地址的后两个字节地址
unsigned char *p4 = (unsigned char*)&a+3; //获取a的首地址的后三个字节地址
printf("[a] p1:%x, %d\r\n", p1, *p1); //打印p1的地址与存储的字节内容
printf("[a] p2:%x, %d\r\n", p2, *p2); //打印p2的地址与存储的字节内容
printf("[a] p3:%x, %d\r\n", p3, *p3); //打印p3的地址与存储的字节内容
printf("[a] p4:%x, %d\r\n", p4, *p4); //打印p4的地址与存储的字节内容

运行结果:

[a] p1:5216f804, 2
[a] p2:5216f805, 1
[a] p3:5216f806, 0
[a] p4:5216f807, 0

可以看出,随着地址的增加,存储的内存依次是2、1、0、0,对应数字258的从底到高的4个字节的值,另外每次运行程序,变量a的地址是自动分配的,所以每次的输出与上面示意图的地址是不同的,但都是4个连续增加的地址值。

另外补充一下,这是一种小端字节序的存储方法,即将一个数据的低字节存储在内存的低地址,或理解为先存储数据的低字节。与之对应的是大端字节序存储方式,即先存储数据的高位字节,类似与我们书写数字时从左到右先写高位数字一样。由于计算机先处理地位字节的效率比较高,因此计算机内部的数据处理均采用小端字节序,而为了方便人的阅读,除了计算机内部处理,其它场合是采用大端字节序的。可通过下面的示意图助记:

另外,需要区分一点,无论大端还小端字节序,在一个字节的内部的8位2进制数,都是按照人类的习惯从左到右存放,如00000010

在字节中也是按照这样存储的,不需要反过来。

C代码实现

分析了这么多,可以编写代码来输出一个数字的2进制格式形式了。具体的代码实现:

void printf_bin(int num)
{int i, j, k;unsigned char *p = (unsigned char*)&num + 3;//p先指向num后面第3个字节的地址,即num的最高位字节地址for (i = 0; i < 4; i++) //依次处理4个字节(32位){j = *(p - i); //取每个字节的首地址,从高位字节到低位字节,即p p-1 p-2 p-3地址处for (int k = 7; k >= 0; k--) //处理每个字节的8个位,注意字节内部的二进制数是按照人的习惯存储!{if (j & (1 << k))//1左移k位,与单前的字节内容j进行或运算,如k=7时,00000000&10000000=0 ->该字节的最高位为0printf("1");elseprintf("0");}printf(" ");//每8位加个空格,方便查看}printf("\r\n");
}

数字258的实际运行效果:

00000000 00000000 00000001 00000010

对应unsigned char型的数字,如果只需要显示8位2进制数,可以对上述小程序简化:

void printf_bin_8(unsigned char num)
{int k;unsigned char *p = (unsigned char*)&num;for (int k = 7; k >= 0; k--) //处理8个位{if (*p & (1 << k))printf("1");elseprintf("0");}printf("\r\n");
}

数字12的运行效果

00001100

关于负数

上面测试的都是正数,当然,对于负数,也是可以显示的,因为负数在计算机中是以对应正数的补码来存储的,因而显示的2进制数也是补码形式,这里顺便再复习一下补码:

对于负数-9,对应的正数位9,9的原码,反码,补码如下:

这里只画出8位示意,int型数字实际是占32位,对应的高位全是0或1。

使用程序测试一下-9的2进制格式输出:

11111111 11111111 11111111 11110111

可以正确显示。

完整测试代码

下面是整个本文的整个测试程序:

#include <stdio.h>
#include <stdlib.h>void printf_bin(int num)
{int i, j, k;unsigned char *p = (unsigned char*)&num + 3;for (i = 0; i < 4; i++) //处理4个字节(32位){j = *(p - i); //取每个字节的首地址for (int k = 7; k >= 0; k--) //处理每个字节的8个位{if (j & (1 << k))printf("1");elseprintf("0");}printf(" ");}printf("\r\n");
}void printf_bin_8(unsigned char num)
{int k;unsigned char *p = (unsigned char*)&num;for (int k = 7; k >= 0; k--) //处理8个位{if (*p & (1 << k))printf("1");elseprintf("0");}printf("\r\n");
}int main()
{int a = 258;int b = -9;printf("定义int a=%d, int b=%d\r\n", a, b);//使用unsigned char来验证int的每一个字节unsigned char *p1 = (unsigned char*)&a;  //获取a的首地址unsigned char *p2 = (unsigned char*)&a+1;//获取a的首地址的后一个字节地址unsigned char *p3 = (unsigned char*)&a+2;//获取a的首地址的后两个字节地址unsigned char *p4 = (unsigned char*)&a+3;//获取a的首地址的后三个字节地址printf("\r\n查看a的每个字节的地址与内容:\r\n");printf("[a] p1:%x, %d\r\n", p1, *p1);//打印p1的地址与存储的字节内容printf("[a] p2:%x, %d\r\n", p2, *p2);//打印p2的地址与存储的字节内容printf("[a] p3:%x, %d\r\n", p3, *p3);//打印p3的地址与存储的字节内容printf("[a] p4:%x, %d\r\n", p4, *p4);//打印p4的地址与存储的字节内容p1 = (unsigned char*)&b;p2 = (unsigned char*)&b + 1;p3 = (unsigned char*)&b + 2;p4 = (unsigned char*)&b + 3;printf("\r\n查看b的每个字节的地址与内容:\r\n");printf("[b] p1:%x, %d\r\n", p1, *p1);printf("[b] p2:%x, %d\r\n", p2, *p2);printf("[b] p3:%x, %d\r\n", p3, *p3);printf("[b] p4:%x, %d\r\n", p4, *p4);//自己的方法1printf("\r\na的2进制格式(显示32位):\r\n");printf_bin(a);printf("\r\nb的2进制格式(显示32位):\r\n");printf_bin(b);//自己的方法2unsigned char c = 12;printf("\r\n定义unsigned char c=%d\r\n", c);printf("c的2进制格式(显示8位):\r\n");printf_bin_8(c);//调用函数的方法printf("\r\n使用_itoa函数显示2进制格式\r\n");char s[32];_itoa(a, s, 2);printf("a --> %s\n", s);_itoa(b, s, 2);printf("b --> %s\n", s);   _itoa(c, s, 2);printf("c --> %s\n", s);getchar();return 0;
}

运行结果:

定义int a=258, int b=-9查看a的每个字节的地址与内容:
[a] p1:a739f4d4, 2
[a] p2:a739f4d5, 1
[a] p3:a739f4d6, 0
[a] p4:a739f4d7, 0查看b的每个字节的地址与内容:
[b] p1:a739f4f4, 247
[b] p2:a739f4f5, 255
[b] p3:a739f4f6, 255
[b] p4:a739f4f7, 255a的2进制格式(显示32位):
00000000 00000000 00000001 00000010b的2进制格式(显示32位):
11111111 11111111 11111111 11110111定义unsigned char c=12
c的2进制格式(显示8位):
00001100使用_itoa函数显示2进制格式
a --> 100000010
b --> 11111111111111111111111111110111
c --> 1100

``

C语言打印数据的二进制格式-原理解析与编程实现相关推荐

  1. c语言输出方框□怎么回事_C语言打印数据的二进制格式-原理解析与编程实现

    问题引出 C语言中,在需要用到16进制数据的时候,可以通过printf函数的%x格式打印数据的16进制形式.在某些位标记.位操作的场合,需要用到2进制格式的数据,但printf函数不能输出2进制格式, ...

  2. mysql新增列并同时增加数据_图解MySQL | [原理解析] MySQL 为表添加列 是怎么quot;立刻quot;完成的...

    在上一期图解 图解MySQL | MySQL DDL为什么成本高?中,我们介绍了: 传统情况下,为表添加列需要对表进行重建 腾讯团队为 MySQL 引入了 Instant Add Column 的方案 ...

  3. 湖仓一体,Hologres加速云数据湖DLF技术原理解析

    Hologres(中文名交互式分析)是阿里云自研的一站式实时数仓,这个云原生系统融合了实时服务和分析大数据的场景,全面兼容PostgreSQL协议并与大数据生态无缝打通,能用同一套数据架构同时支持实时 ...

  4. 用linux编译打印杨辉三角形,C语言打印杨辉三角代码及解析

    杨辉三角是我们从初中就知道的,现在,让我们用C语言将它在计算机上显示出来. 在初中,我们就知道,杨辉三角的两个腰边的数都是1,其它位置的数都是上顶上两个数之和.这就是我们用C语言写杨辉三角的关键之一. ...

  5. mysql 二进制格式_解析MYSQL BINLOG 二进制格式(3)--QUERY_EVENT

    该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 FORMAT_DESCRIPTION_EVENT class:Query_log_event event:QUERY_EVENT event_code:0 ...

  6. java语言打印1到10的偶数_#Java编程# 编写一个应用程序创建两个线程,一个线程打印输出1~100之间所有的奇数,另外一 求写一个用JAVA求1~...

    import java.util.Random; class A extends Thread { \tint i=1; \tRandom r=new Random(); \tpublic void ...

  7. C语言实现十进制转二进制

    C语言实现十进制转二进制 数学原理 用2整除十进制整数,可以得到一个商和余数:再用2去除商,又会得到一个商和余数,如此进行,直到商为小于1时为止,然后把先得到的余数作为二进制数的低位有效位,后得到的余 ...

  8. [matlab编程实践].cli文件二进制格式读取【3D打印切片文件】

    文章目录 零.cli文件简介 0.0 cli文件说明文档 0.1 下面是我们这使用的3D打印机的流程. 0.2 cli文件格式 0.3 cli文件结构 0.4 几何数据部分详解 一.matlab 程序 ...

  9. 计算机中十进制转二进制逻辑原理,.计算机中为什么要采用二进制?及二进制的基本运算规则,还有.二进制数据与十进制、八进制、十六进制数据之间的转换方法?...

    满意答案 sqq212527 2013.04.11 采纳率:46%    等级:12 已帮助:10556人 编辑本段简介 20世纪被称作第三次科技革命的重要标志之一的计算机的发明与应用,其运算模式正是 ...

  10. 大规模分布式存储系统:原理解析与架构实战 (大数据技术丛书) - 电子书下载 -(百度网盘 高清版PDF格式)...

    大规模分布式存储系统:原理解析与架构实战 (大数据技术丛书)-杨传辉 在线阅读                   百度网盘下载(89hy) 书名:大规模分布式存储系统:原理解析与架构实战 (大数据技 ...

最新文章

  1. 计算机视觉的数据增广技术大盘点!附涨点神器,已开源!
  2. 各种好用的代码生成器(C#)
  3. 主机无法访问虚拟机的httpd服务
  4. Part7 继承与派生 7.4派生类的构造和析构
  5. java jdbc 工具_JDBC 工具类
  6. python3.7安装opencv4.1_树莓派4的opencv4.1.0--python3.7的开发环境安装
  7. 安防摄像头WEB端直播,实现按需播放,节省带宽和服务器压力
  8. apple id两步验证服务器,什么是Apple ID两步验证 苹果ID二次验证流程
  9. 机械制图国家标准的绘图模板_JS制图:映射
  10. java面向对象编程培训_Java面向对象编程简介
  11. 【计算机网络】集线器、网桥、交换机、路由器、网关大解析
  12. Win10系统双击无法打开图片的解决方法
  13. ERC20重要补充之approveAndCall
  14. 什么因素影响阿里云国际版云服务器延迟?
  15. bugku-misc练习(一)
  16. Element表格内容不垂直居中和内容换行显示
  17. Java中String 字符串与List<String>互转
  18. 分治法:找出不合格银币
  19. html页面控制标签,html常用标签大全
  20. 当前速度火车测试 软件,四大主流火车购票APP对比测评教程

热门文章

  1. 计算机放音乐声音小在吗调,用电脑在网上看电影时配乐音量太大,请问怎么只把配乐音量调小,而说话声不调小。...
  2. 1415-2结对分组情况
  3. 如何搭建DNS--域名系统
  4. word里面的ctrl+z快捷键无效|ctrl+z 无效 最新解决方法
  5. pdf转换器中文版下载
  6. 软件测试应遵循哪些原则?
  7. 各大电商平台竞价比价API,实时监控类目大数据
  8. 魅族7.0系统最简单激活Xposed框架的经验
  9. spring系统学习之控制反转 ioc
  10. 优秀的第二外语学习网站:Lang-8