2012年软考软件设计师辅导:C语言编程常见BUG

来源:微学教育网发布时间:2012-04-27

  记录一些C语言易出的bug,有些是自己编程中遇到的,有些是在其他地方看到的:

  1、释放动态内存时遇到“user breakpoint called from code at 0x********”

  这个bug多在调用free函数来释放malloc的内存时出现,前先曾多次遇到,但一直未找出问题原因。

  在网上查了一下,有些写得挺复杂,涉及到了操作系统推维护的内容,现在还不有接触到这一块,所以没有细看。其他有说到重复释放动态内存,修改动态内存指针或者破坏了系统的动态内存结束标志都会导致这个bug。后来查看自己的代码,发现重复释放的问题倒是没有,却在操作内存时有一处频繁操作超出动态申请内存之外的单元,故导致这个问题的出现。

  问题总结:在动态内存释放时出现,多由于动态内存使用不当

  1、重复释放动态内存

  2、读写操作超出了动态申请内存边界(读应该不会发生问题,但也要避免,除非你真正明白自己在干什么)

  2、C中的字符串异常

  严格地说,C中是没有字符串变量的,一般采用字符数组或字符指针的方式才实现字符串的操作,二者在使用中有诸多类似,但并不是完全等价的,注意下面两句的区别:

  char a[] = “Hello world”;//存放在栈中,可以修改a数据组的任意值

  char *s = “Hello world”;//指针s存放在栈中,字符串常量在代码段,不可修改

  给s[i]赋值将导致错误

  3、条件判断时一定要学会利用短路特性来防止异常,如

  while(j >=0 && a[j] >0) j--;

  if(d != 0 && n/d == 0)

  if(p == NULL || *p == ’\0‘)/*no string*/

  例中3种情况或将两个判断条件调换了次序都会导致内存溢出。

  4、关于混乱的类型扩展问题

  因为不确定编译器采用哪种保护规则(无符号保护或是值保护),应尽量避免在同一个表达式内混用有符号和无符号的变量。任何时候,总可以用显式的类型转换来明确无误地表达所希望的转换的地方或方式。

  5、类型转换被当成左值

  在C语言中,类型转换操作只能生成一个右值,不能被赋值或进行自增(减)运算。

  如

  char *p;

  ((int *)p)++;

  不能完成预想中的将p增加一个int的长度,而应该如下:

  p= (char *)((int *)p + 1);

  或p += sizeof(int);

  6、变长参数函数调用中,会进行默认类型转换吗?

  下面这段代码的输出是什么?

  #include <stdio.h>

  int main()

  {

  __int64 n;

  int b;

  n = 1;

  b = 2;

  printf(“%d,%d\n”,n,b);

  return 0;

  }

  和预料不同的是,它输出:1,0

  这是因为在可变参数函数调用中,一般编译器是不进行默认类型转换的,当函数根据格式字符串从栈中提取参数时就会由于变量长度不一致而产生异常的输出。