在C语言中, signed char 类型的范围为-128~127,每本教科书上也这么写,但是没有哪一本书上(包括老师)也不会给你为什么是-128~127,这个问题貌似看起来也很简单容易, 以至于不用去思考为什么,不是有一个整型范围的公式吗:  -2^(n-1)~2^(n-1)-1   n为整型的内存占用位数,所以int类型32位 那么就是 -(2^31)~2^31 -1 即

-2147483648~2147483647,但是为什么最小负数绝对值总比最大正数多1 ,这个问题甚至有的工作几年的程序员都模棱两可,因为没有深入思考过,只知道书上这么写。。于是,我不得不深入思考一下这个被许多人忽视的问题。。

对于无符号整数,很简单,全部位都表示数值,比如 char型,8位,用二进制表示为0000 0000 ~ 1111 1111

1111 1111 最大即为十进制255,所以 unsigned char 的范围为0~ 255,在这里普及一下2进制转十进制的方法, 二进制每一位的数值乘以它的位权(2^(n-1),n为自右向左的位),再相加,可得到十进制数,比如 :

1111 1111 =1*2^7+1*2^6+1*2^5+1*2^4+1*2^3+1*2^2+1*2^1+1*2^0=127  。

但是对于有符号整数,二进制的最高位表示正负,不表示数值,最高位为0时表示正数,为1时表示负数,这样一来,能表示数值的就剩下(n-1)位了,比如 char a= -1;   那么二进制表示就为 1 0000001,  1 表示为0 0000001 ,所以signed char 型除去符号位剩下的7位最大为1111 111 =127,再把符号加上,0 1111111=127 ,1 1111111= -127,范围应该为 -127~127 ,同理int类型也一样,但是问题出来了,教科书上是-128~127 啊,下面就剖析一下这个惊人的奇葩。。。

再普及一下计算机内部整数存储形式,大家都知道计算机内部是以二进制来存贮数值的,无符号整数会用全部为来存储,有符号的整数,最高位当做符号位 ,其余为表示数值,这样貌似合理, 却带来一个麻烦,当进行加法时,1+1

0000 0001

+     0000 0001

—————————

0000  0010    ………………2

当相减时 1-1=?  由于计算机只会加法不会减法,它会转化为1+(-1) ,因此

0000 0001

+    1000 0001

____________________

1000 0010     …………… -2    ,1-1= -2?  这显然是不对了,所以为了避免减法运算错误,计算机大神们发明出了反码,直接用最高位表示符号位的叫做原码, 上面提到的二进制都是原码形式,反码是原码除最高位其余位取反,规定:正数的反码和原码相同,负数的反码是原码除了符号位,其余为都取反,因此-1 的源码为 1 0000001 ,反码为 1 1111110, 现在再用反码来计算 1+(-1)

0000 0001

+    1111 1110

————————

1111 1111       …………再转化为原码就是 1000 0000 = -0  ,虽然反码解决了相减的问题,却又带来一个问题,-0 ,既然0000 0000 表示 0,那么就没有 -0 的必要, 出现 +0= -0=0 ,一个0 就够了,为了避免两个0的问题,计算机大师们又发明了补码,补码规定: 整数的补码是其本身,负数的补码为其反码加一 ,所以,负数转化为反码需两个步骤, 第一,先转化为反码,第二: 把反码加一。。这样 -1 的补码为 1111 1111    ,1+(-1)

0000 0001

+   1111 1111

________________

1  0000 0000  ……………………  这里变成了9位,由于char 为8位,最高位1 被丢弃 结果为0 ,运算正确

再看, -0 :原码 1000 0000 的补码为1 0000 0000 ,由于char 是 八位 ,所以取低八位00000000,   +0 :原码为0000 00000 ,补码为也为 0000 0000 ,虽然补码0都是相同的,但是有两个0 ,既然有两个0 ,况且0既不是正数,也不是负数, 用原码为0000 0000 表示就行了, 这样一来,有符号的char ,原码都用来表示-127~127 之间的数了,唯独剩下原码1000 0000 没有用,用排列组合也可以算出来,0???????,能表示2^7=128个数,刚好是0~127, 1???????,也能表示128个数,总共signed char 有256 个数,这与-127~127 中间是两个0 刚好吻合。。现在再来探讨一下关于剩下的那个1000 0000,

既然-127 ~0~ 127都有相应的原码与其对应,那么1000 0000 表示什么呢,当然是-128了,为什么是-128呢,网上有人说-0即1000 0000 与128的补码相同,所以用1000 0000表示-128,,这我实在是不敢苟同,或者说-128没有原码,只有补码1000 0000,胡扯,既然没有原码何来补码,还有说-128的原码与-0(1000 0000)的原码相同,所以可以用1000 0000表示-128,我只能说,回答的不要那么牵强, 原码1000 0000 与-128的原码实际上是不同的, 但为什么能用它表示-128进行运算,如果不要限制为char 型(即不要限定是8位),再来看,-128的原码:1 1000 0000 ,9位,最高位符号位,再算它的反码:1 0111 1111,进而,补码为: 1 1000 0000,这是-128的补码,发现和原码一样, 1 1000 0000和1000 0000 相同?如果说一样的人真是瞎了眼了,所以,-128的原码和-0(1000 000)的原码是不同的,但是在char 型中,是可以用1000 000 表示-128的,关键在于char 是8位,它把-128的最高位符号位1 丢弃了,截断后-128的原码为1000 000 和-0的原码相同,也就是说

1000 0000  和-128丢弃最高位后余下的8位相同,所以才可以用-0 表示-128,这样,当初剩余的-0(1000 0000),被拿来表示截断后的-128,因为即使截断后的-128和char 型范围的其他数(-127~127)运算也不会影响结果, 所以才敢这么表示-128。

比如 -128+(-1)

1000 0000  ------------------丢弃最高位的-128

+     1111  1111   -----------------   -1

________________

10111  1111    ------------------char 取八位,这样结果不正确,不过没关系 ,结果-129本来就超出char型了,当然不能表示了。

比如 -128+127

1000 0000

+ 0111 1111

————————

1111 1111 --------------  -1 结果正确, 所以,这就是为什么能用 1000 0000表示-128的原因。

从而也是为什么char 是-128~127,而不是-127~127 ,short int 同样如此 -32768~32767  因为在16位中,-32768为原码为17位,丢弃最高位剩下的16为- 0 的原码相同。。。。

还有一个问题:

既然-128最高位丢弃了。那么

char a=-128;  //在内存中以补码1 1000 0000 存储,但由于是char ,所以只存储 1000 0000

printf("%d",a); //既然最高位丢弃了,输出时应该是1000 000 的原码的十进制数-0 ,但为什么能输出-128呢。

还能打印出-128;

我猜想是计算机内部的一个约定,就像float一样 ,能用23位表示24位的精度 ,因为最高位默认为1,到时候把23位取出再加 1便可。

-128也是同样的原理,当数据总线从内存中取出的是1000 000 ,CPU会给它再添最高一位,变为1 1000 0000 这样才能转化为

-128输出,不然1000 0000 如何输出?这当然是我的一种推断,具体怎么实现还得问CPU的设计者了。。。。

再看一个例子:

char a=-129;

printf("%d",a)  ;    会输入多少??    结果为127 ,为什么呢?

-129在补码为10 0111 1111 只取后八位存储,即 0111 111 这个值刚好是127了,同理-130 截断后为126.....

如此按模轮回,关于模就先不探讨了。。

那么

unsigned  char a=  -1;

if( 1>a)   printf("大于");

else

printf("小于");

结果是什么呢?  出人意料的是:  小于,而不是大于,猫腻在你哪呢,还是存储问题:

a为unsigned 无符号, 它的八位都用来存储数值, 没有符号位,编译器把 -1 转换为补码为 1111 1111,但由于是无符号,计算机会把 1111 11111 当做是无符号来对待 ,自然就是 2^8 -1  = 255 了,所以相当于是if( 1>255) 肯定是

printf("小于");了。。。

。。。。。。。。。。。

好了,就说到这儿吧。。。。。。。。

————————————————
版权声明:本文为CSDN博主「若水三千你是一千」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/daiyutage/article/details/8575248

浅析为什么char类型的范围是 —128~+127相关推荐

  1. 8位alu运算器(vhdl语言)_C语言中signed char类型,能表示-128到127,为什么负数多一位?...

    初学者在学习C语言,谈到不同数据类型时,一般都能理解 unsigned 和 signed 的区别,无非就是有无符号而已.但是对于 signed 数据类型的数据范围,初学者却常常会感到迷惑. 对于 si ...

  2. c语言signed char,C语言 1字节signed char的范围为什么是-128~127?

    参考 无符号单字节范围 无符号单字节unsigned char位宽是8bit,范围[0, 1111 1111b],也就是[0, 255]. 有符号单字节范围 有符号单字节signed char位宽也是 ...

  3. 浅谈char类型范围

    在 C 语言中, signed char 类型的范围为 -128~127,每本教科书上也这么写,但是没有哪一本书上(包括老师)也不会给你为什么是 -128~127,这个问题貌似看起来也很简单容易, 以 ...

  4. char类型的大小范围

    要知道char类型的大小范围,首先要知道char类型占1个字节,而1个字节等于8个bit位,每一个bit位是一位二进制数,所以char类型有8个二进制位,那么就有2的8次方256种可能. 我们先讨论( ...

  5. 位操作符左移问题(<<)如果char类型左移后位数超出一字节如何打印?

    目录 有符号char 现在有两个问题: 为什么是256? 为什么是负数 无符号char 为探究该问题,我们暂且设char sh = 128 有符号char 打印后变成了负数-256 现在有两个问题: ...

  6. c语言中char的取值扩大,C语言中 char 类型的取值范围为什么是-128~127

    我们之前已经说过关于原码.反码和补码的一些东西,如果你没有看过,可以点这里< 你知道原码.反码和补码吗,进来了解一下吧 >看一下 . 好了,可能你不会太想看,所以我们一起再来简单的复习一下 ...

  7. C/C++ unsigned char*类型

    C++ unsigned char *是表示无符号字符指针的意思. 细节如下: char 前面添加unsigned表示是无符号的字符,也就是不可以存储负数: 在数据类型后面加*表示指针的意思: 指针是 ...

  8. char类型是多少 mat_OpenCV Mat数据类型及位数总结

    OpenCV Mat数据类型及位数总结 float:  4字节,6-7位有效数字 -3.4E-38 到 3.4E38 double: 8字节,15~16位有效数字 -1.7E-308 到 1.7E30 ...

  9. C++中char类型的溢出问题

    C++中什么经常会运用到char类型,也会将char类型作为循环语句的循环条件,但往往这里最容易出现错误,容易出现溢出,进入死循环.这里我们就来简单介绍下为什么会出现这种情况. 首先,了解下char类 ...

最新文章

  1. 特别推荐:15个精美 Metro UI 风格 WordPress 主题
  2. 用移位寄存器实现边沿检测(功能仿真及ISE综合)
  3. bootstrap-警告框中的链接
  4. deepin下载软件慢切换镜像
  5. android gradle 目录,Android Gradle:将目录列入文件
  6. opencv中的createsamples.exe生成vec文件注意事项
  7. linux+不同分区mv,mv操作深入浅出
  8. include函数_include()函数以及JavaScript中的示例
  9. no.7_qzhai 开心版_开心宝贝GM版下载-开心宝贝GM版安卓下载
  10. python中sys模块是什么意思_python之sys模块详解
  11. SAP License:SAP顾问是如何炼成的——我所理解的SAP顾问
  12. 如何使用 vimdiff 来 git diff /svn diff
  13. 成为0.01%!利用TensorFlow.js和深度学习,轻松阅读古草体文字
  14. php 图像生成缩略图
  15. 第五章 政策问题与议程设定
  16. ubuntu中各个文件夹的作用
  17. 梁山伯与祝英台最动人的演绎
  18. MySQL系列——MySQL实现序列(Sequence)效果
  19. github 修改密码密码_雇主要求提供GitHub密码。 如何处理呢?
  20. icloud备份qq数据怎么恢复

热门文章

  1. 二十四节气和计算机专业,墨迹万年历专业日历“二十四节气”带你行走不一样的2021...
  2. 超级眼电脑监控软件保护企业机密信息让商业间谍无处遁形
  3. 关于主板PCIE对硬盘速度影响的思考(B450F)
  4. 华为荣耀v8计算机没了,华为荣耀V8失败变砖开不了机了怎么办_华为荣耀V8救砖方法...
  5. picgo-plugin-imageX火山引擎ImageX插件for picGo
  6. 算法设计与分析——淘汰赛冠军问题(Java)
  7. 博客升级_原水_新浪博客
  8. css3绘制的钢铁侠代码
  9. 全球中文论坛百强排行榜完全名单
  10. 如何建一个小型服务器存文件,小型文件服务器