Chinaunix首页 | 论坛 | 博客
  • 博客访问: 429711
  • 博文数量: 113
  • 博客积分: 2228
  • 博客等级: 大尉
  • 技术积分: 1341
  • 用 户 组: 普通用户
  • 注册时间: 2012-10-24 15:07
文章分类

全部博文(113)

文章存档

2013年(52)

2012年(61)

我的朋友

分类: C/C++

2013-02-26 14:37:29

《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 
void main() 
{int a[5],i=0; 
for(i=0;i<5;i++) 
{a[i]=i; 
printf("%d\n",*(a+i)); 



17. 以下标的形式访问指针; 
如:#include 
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 
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)语
句校验将不起作用。 

阅读(1210) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~