请看下面这段代码:
#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)只是一个随机值。
推荐阅读
- C语言学习|第十一届蓝桥杯省赛 大学B组 C/C++ 第一场
- 【C】题目|【C语言】题集 of ⑥
- 单片机|自学单片机好找工作吗(会单片机能找什么工作?)
- 单片机|keil把源代码生成lib的方法
- c语言|一文搞懂栈(stack)、堆(heap)、单片机裸机内存管理malloc
- c语言|C语言初期学习遇到的特殊点 【三子棋详解】【初学者福音,详细总结,复习能手】
- 笔记|C语言数据结构——二叉树的顺序存储和二叉树的遍历
- 个人理解|【C语言基础之类型转换】
- c语言|【C语言】自定义类型 结构体 枚举 联合
- 学习分享|【C语言函数基础】