C语言|陷阱之有符号与无符号

请看下面这段代码:

#include #include int main(int argc, char **argv) { char a[1000]; int i; for(i = 0; i < 1000; i++) { a[i] = -1 - i; } printf("%c\n",a[190]); 1 printf("%c\n",a[254]);2 printf("%c\n",1); 3 printf("%d\n",strlen(a)); printf("%d\n",i); printf("%d\n",a[i]); return 0; }

【C语言|陷阱之有符号与无符号】打印值:
255
1000
-24
对于这个结果大家一定很困惑吧,可能只有中间一个结果大家还能接受。
这里主要用到了补码的知识。
首先为什么第一个值是255呢?
首先a[0]的值是-1,关键就是-1在内存中是如何存储的呢?我们知道在计算机中,数值一律用补码来表示,主要原因是使用补码,可以将符号位和其他位统一处理;同时,减法也可按照加法来处理。另外,两个补码表示的数相加时,如果高位有进位,则进位被舍弃。
补码的规则:
正数的补码与其原码一致,负数的补码符号位(最高位)为1,其余位为该数绝对值的原码按位取反,然后再加1。
按照补码的规则,可以知道-1的补码是0xff,-2的补码是0xfe……当i的值为127时,a[127]的值是-128,而-128是char类型数据能表示的最小的负数。当i继续增加,a[128]的值肯定不能是-129。因为这时候发生了溢出,-129需要9位才能存储下来,而char类型数据只有8位,所以最高位被舍弃。剩下的8位是9位补码的低8位的值,即0x7f。当i继续增加到255的时候,-256的补码的低8位为0。所以从a[0]到a[254]的值都不为0,而a[255]刚好为字符0(即\0),所以strlen的长度是从a[0]到a[254],结果为255。
而最后那个a[i](i = 1000)只是一个随机值。

    推荐阅读