C语言深度剖析笔记
1.在c语言中,凡不加返回值类型限定的函数,就会被编译器作为返回整形处理。 2.register 变量必须是一个单个的值,并且其长度应小于或等于整型的长度。而且 register 变量可能不存放在内存中, 所以不能用取址运算符 “&”来获取 register变量的地址。 3. int main() { char a[1000]; int i; for(i=0; i<1000; i++) { a[i] = -1-i; } printf("%d",strlen(a)); return 0; } 此题看上去真的很简单,但是却鲜有人答对。答案是 255。别惊讶,我们先分析分析。for循环内,当 i 的值为 0时,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。然后当i增加到 256时,-257的补码的低 8 位全为1,即低八位的补码为 0xff,如此又开始一轮新的循环……按照上面的分析,a[0]到 a[254]里面的值都不为 0,而 a[255]的值为 0。strlen函数是计算字符串长度的,并不包含字符串最后的‘\0’ 。而判断一个字符串是否结束的标志就是看是否遇到‘\0’ 。如果遇到‘\0’ ,则认为本字符串结束。分析到这里,strlen(a)的值为 255应该完全能理解了。这个问题的关键就是要明白 char类型默认情况下是有符号的,其表示的值的范围为[-128,127],超出这个范围的值会产生溢出。 另外还要清楚的就是负数的补码怎么表示。 弄明白了这两点, 这个问题其实就很简单了。 4.case 后面只能是整型或字符型的常量或常量表达式(想想字符型数据在内存里是怎么存的) 。 5. 不能对 void指针进行算法操作。 6. return 语句不可返回指向“栈内存”的“指针” ,因为该内存在函数体结束时被自动销毁。 7. struct student {}stu; sizeof(stu)的值是多少呢?是1。即空结构体的大小就定位 1 个byte。 8. 编译器会将注释剔除,但不是简单的剔除,而是用空格代替原来的注释。 9. 注意:/*…*/这种形式的注释不能嵌套,因为/*总是与离它最近的*/匹配。 10.const修饰的只读变量不能用来作为定义数组的维数, 也不能放在 case关键字后面。 11. C语言里(\)表示断行。以反斜杠反斜杠之后不能有空格,反斜杠的下一行之前也不能有空格。 12. 注释先于预处理指令被处理。因此,试图用宏开始或结束一段 注释是不行的。 13.在32位系统下,不管什么样的指针类型,其大小都为 4byte。 14. 往内存 0x12ff7c地址上存入一个整型数 0x100,可以用下面的方法:int *p = (int *)0x12ff7c; *p = 0x100; 或者直接这么写代码:*(int *)0x12ff7c = 0x100; 15. int a[5];a作为右值时,代表数组首元素的首地址,而非数组的首地址。 16.以指针的形式访问数组; 如:#include<stdio.h> void main() {int a[5],i=0; for(i=0;i<5;i++) {a[i]=i; printf("%d\n",*(a+i)); } } 17. 以下标的形式访问指针; 如:#include<stdio.h> void main() {int a[5],i=0,*p; p=a; for(i=0;i<5;i++) {a[i]=i; printf("%d, ",p[i]); } } 18. 对指针进行加1操作,得到的是下一个元素的地址,一个类型为T的指针的移动,以 sizeof(T) 为移动单位。 例:#include<stdio.h> void main() {int a[5]={1,2,3,4,5}; int *ptr=(int *)(&a+1); printf("%d,%d",*(a+1),*(ptr-1)); } 19. 函数本身是没有类型的,只有函数的返回值才有类型。 20. malloc函数的原型:(void *)malloc(int size)。malloc函数的返回值是一个 void类型的指针,参数为 int 类型数据。内存分配成功之后,malloc函数返回这块内存的首地址。你需要一个指针来接收这个地址。malloc函数申请的是连续的一块内存。注意malloc函数申请内存有可能不成功,所以我们在使用指向这块内存的指针时,必须用 if(NULL !=p)语句来验证内存确实分配成功了。 注意:用 malloc函数申请 0字节时,内存不会返回 NULL指针,而是返回一个正常的内存地址。但是你却无法使用这块大小为 0 的内存。对于这一点一定要小心,因为这时候 if(NULL !=p)语句校验将不起作用。