关于补码。C语言

Citron09 2016-09-24 07:10:59
好久的事了。想确认下。
mian函数以前都喜欢return 0或者1,测试失败还是成功,有一次无意间返回了一次-1;
返现编译器的报告:
当时想了想,-1的补码不就是0xffffffff么,也没在意,觉得可能是直接输出的计算机存储的值。后来无聊。测试了一个结果
int nTest = 0xffffffff;
std::cout << nTest;
,结果是-1,为什么。我知道计算机是用补码存储数据的,但是补码输出难道不转换回来而是直接输出么。
然后想到了可能是溢出了,因为int在我计算机中是32位。除去符号位:2 147 483 648,0xffffffff出去符号位不也是31位么。
为什么?讨论了下得出的结果是存储16进制数计算机会输出的时候直接输出补码,不大相信。。。
...全文
415 10 打赏 收藏 转发到动态 举报
写回复
用AI写文章
10 条回复
切换为时间正序
请发表友善的回复…
发表回复
Citron09 2016-09-25
  • 打赏
  • 举报
回复
引用 6 楼 fefe82 的回复:
[quote=引用 5 楼 Citron09 的回复:] [quote=引用 4 楼 fefe82 的回复:] [quote=引用 3 楼 paschen 的回复:] -1 补码与原码是一样的
可不一样 ...
-1 的
原码: 0x80000001
反码: 0xfffffffe
补码: 0xffffffff
[/quote] 你是说在int nTest = 0xffffffff的时候,若是没有发生溢出,比如说long int lTest=0x80000000ffffffff;,它在内存里面存放的话就是存放他的补码,但是现在发生溢出了,因此直接将0xffffffff作为补码存放进去了么,是这样理解么?[/quote] long int lTest=0x80000000ffffffff 这个也溢出了吧 ... 放进内存里的东西和你写的东西是两回事。只不过对以整数来说他们碰巧经常是一样的而已。 编译器在把一个数放进内存之前,要先确定他的数值,跟他的类型。 之后,根据类型对数值进行编码。(对有符号整数来说,经常是补码) 0xffffffff 数值是 2^32-1 ,类型是 unsigned int 。把他赋给 int ,要经过类型转换(implementation-defined, 这里溢出了),结果数值是 -1 ,类型 int 。之后,写进内存的是按照 int (补码)编码的 -1 。输出的时候,也是按照 nTest 的类型 (int) 解释内存中的内容,进行输出。 你可以想一下 double dTest = 0xffffffff; 会发生什么。[/quote] 也就是说,还是因为类型转换产生了溢出,编译器处理溢出后,转换为了-1是么?请问您是怎么知道这个的。。。有什么参考文献么我想看看
赵4老师 2016-09-25
  • 打赏
  • 举报
回复
电脑内存或文件内容或传输内容只是一个一维二进制字节数组及其对应的二进制地址; 人脑才将电脑内存或文件内容或传输内容中的这个一维二进制字节数组及其对应的二进制地址的某些部分看成是整数、有符号数/无符号数、浮点数、复数、英文字母、阿拉伯数字、中文/韩文/法文……字符/字符串、汇编指令、函数、函数参数、堆、栈、数组、指针、数组指针、指针数组、数组的数组、指针的指针、二维数组、字符点阵、字符笔画的坐标、黑白二值图片、灰度图片、彩色图片、录音、视频、指纹信息、身份证信息……
paschen 版主 2016-09-25
  • 打赏
  • 举报
回复
引用 4 楼 fefe82的回复:
[quote=引用 3 楼 paschen 的回复:] -1 补码与原码是一样的
可不一样 ...
-1 的
原码: 0x80000001
反码: 0xfffffffe
补码: 0xffffffff
[/quote] 是的,正数才是一样的,昨天不知怎么想的。。
fefe82 2016-09-25
  • 打赏
  • 举报
回复
引用 9 楼 Citron09 的回复:
[quote=引用 6 楼 fefe82 的回复:] [quote=引用 5 楼 Citron09 的回复:] [quote=引用 4 楼 fefe82 的回复:] [quote=引用 3 楼 paschen 的回复:] -1 补码与原码是一样的
可不一样 ...
-1 的
原码: 0x80000001
反码: 0xfffffffe
补码: 0xffffffff
[/quote] 你是说在int nTest = 0xffffffff的时候,若是没有发生溢出,比如说long int lTest=0x80000000ffffffff;,它在内存里面存放的话就是存放他的补码,但是现在发生溢出了,因此直接将0xffffffff作为补码存放进去了么,是这样理解么?[/quote] long int lTest=0x80000000ffffffff 这个也溢出了吧 ... 放进内存里的东西和你写的东西是两回事。只不过对以整数来说他们碰巧经常是一样的而已。 编译器在把一个数放进内存之前,要先确定他的数值,跟他的类型。 之后,根据类型对数值进行编码。(对有符号整数来说,经常是补码) 0xffffffff 数值是 2^32-1 ,类型是 unsigned int 。把他赋给 int ,要经过类型转换(implementation-defined, 这里溢出了),结果数值是 -1 ,类型 int 。之后,写进内存的是按照 int (补码)编码的 -1 。输出的时候,也是按照 nTest 的类型 (int) 解释内存中的内容,进行输出。 你可以想一下 double dTest = 0xffffffff; 会发生什么。[/quote] 也就是说,还是因为类型转换产生了溢出,编译器处理溢出后,转换为了-1是么?请问您是怎么知道这个的。。。有什么参考文献么我想看看[/quote] C++ 标准 https://isocpp.org/std/the-standard http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2014/n4296.pdf 不过这东西可能不太适合初学者
fefe82 2016-09-24
  • 打赏
  • 举报
回复
引用 5 楼 Citron09 的回复:
[quote=引用 4 楼 fefe82 的回复:] [quote=引用 3 楼 paschen 的回复:] -1 补码与原码是一样的
可不一样 ...
-1 的
原码: 0x80000001
反码: 0xfffffffe
补码: 0xffffffff
[/quote] 你是说在int nTest = 0xffffffff的时候,若是没有发生溢出,比如说long int lTest=0x80000000ffffffff;,它在内存里面存放的话就是存放他的补码,但是现在发生溢出了,因此直接将0xffffffff作为补码存放进去了么,是这样理解么?[/quote] long int lTest=0x80000000ffffffff 这个也溢出了吧 ... 放进内存里的东西和你写的东西是两回事。只不过对以整数来说他们碰巧经常是一样的而已。 编译器在把一个数放进内存之前,要先确定他的数值,跟他的类型。 之后,根据类型对数值进行编码。(对有符号整数来说,经常是补码) 0xffffffff 数值是 2^32-1 ,类型是 unsigned int 。把他赋给 int ,要经过类型转换(implementation-defined, 这里溢出了),结果数值是 -1 ,类型 int 。之后,写进内存的是按照 int (补码)编码的 -1 。输出的时候,也是按照 nTest 的类型 (int) 解释内存中的内容,进行输出。 你可以想一下 double dTest = 0xffffffff; 会发生什么。
Citron09 2016-09-24
  • 打赏
  • 举报
回复
引用 4 楼 fefe82 的回复:
[quote=引用 3 楼 paschen 的回复:] -1 补码与原码是一样的
可不一样 ...
-1 的
原码: 0x80000001
反码: 0xfffffffe
补码: 0xffffffff
[/quote] 你是说在int nTest = 0xffffffff的时候,若是没有发生溢出,比如说long int lTest=0x80000000ffffffff;,它在内存里面存放的话就是存放他的补码,但是现在发生溢出了,因此直接将0xffffffff作为补码存放进去了么,是这样理解么?
fefe82 2016-09-24
  • 打赏
  • 举报
回复
引用 3 楼 paschen 的回复:
-1 补码与原码是一样的
可不一样 ...
-1 的
原码: 0x80000001
反码: 0xfffffffe
补码: 0xffffffff
paschen 版主 2016-09-24
  • 打赏
  • 举报
回复
-1 补码与原码是一样的
paschen 版主 2016-09-24
  • 打赏
  • 举报
回复
0xFFFFFFFF转换回来就是-1啊,所以输出的是-1没错啊
fefe82 2016-09-24
  • 打赏
  • 举报
回复
1) 关于补码 32bit 下,0xfffffff 就是 -1 。 2) 关于 C++ 0xfffffff 值为 2^32-1,由于超过的 int 的表示范围,但是在 unsigned int 表示范围之内,其类型为 unsigned int, 值为 2^32-1 。 用它初始化一个 int 的是时候(int nTest = 0xffffffff;),会发生转换。 当原数据不在 int 表示范围之内的时候,转换的方式有编译器确定。 对于现在多数用补码表示整数的编译器来说,会直接将二进制表示作为补码解释,结果为 -1 。 这一转换在初始化的时候已经完成了,nTest 从始至终都是 -1 。 ============= c++ 里,数据类型决定了内存中数据将被如何解释。内存中的内容相同,但是类型不同,解释是不一样的。(比如,按不同数据类型解释的数值是不同的)。 在不同数据类型的数据进行赋值的时候,会发生类型的转换。这一转换并不总是将内存中的数据用目标类型重新解释一下。

64,654

社区成员

发帖
与我相关
我的任务
社区描述
C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下

试试用AI创作助手写篇文章吧