点击上方,选择“置顶/星标公众号”
福利干货,第一时间送达 C语言是一种非常流行的编程语言,因为它简单易学,且广泛应用于各个领域。但是,由于C语言本身的特性,它也容易引起一些错误和陷阱,这些错误可能导致程序崩溃、数据丢失或者安全漏洞等问题。本文将介绍15个常见的C语言陷阱,并给出相应的解决方法。
结尾有一道面试题,欢迎大家讨论学习
C语言中有许多运算符,例如加减乘除、逻辑运算符等等。在表达式中,不同运算符的优先级不同,如果没有注意到这一点,就会产生一些错误。例如:
int a = 5, b = 3;
int c = a++ * --b; // a = 6, b = 2以及c = 10。
这个例子中,和--的优先级高于*,所以a和--b先被执行,然后才是乘法运算。如果把上面的代码写成下面这样,结果就会完全不同:
int a = 5, b = 3;
int c = ++a * b--; // 此时a=6,b=2,c=18
此时++a和b--先被执行,然后才是乘法运算。
解决方法:正确理解各个运算符的优先级,并使用括号来明确表达式中各个部分的计算顺序。
在C语言中,变量名和函数名是大小写敏感的。也就是说,myVar和MyVar是两个不同的变量名。这很容易引起混淆和错误,例如:
int MyVar = 5;
int myvar = 3;
printf("%d\n", MyVar + myvar); // 输出8
解决方法:保持一致性,使用统一的命名规则来避免混淆。
数组越界是指访问数组时超出了数组边界的范围。这种情况可能导致程序崩溃或者数据被破坏。例如:
int arr[3] = {1, 2, 3};
int x = arr[3]; // 访问越界
解决方法:注意数组的边界范围,避免访问超出范围的元素。
在C语言中,整型溢出是一个常见的问题。当一个整数超出了它所能表示的范围时,它的值会发生“环绕”,即从最大值变成最小值,或者从最小值变成最大值。例如:
unsigned char x = 255;
x += 1; // 此时x的值为0
解决方法:使用合适的数据类型,避免超出它们所能表示的范围。
指针是C语言中的一个重要概念,但也容易引起一些错误。例如,当一个指针被赋值为NULL后,如果没有判断就继续使用它,就会产生一些奇怪的结果:
int *p = NULL;
*p = 5; // 错误:访问空指针
解决方法:在使用指针之前,检查它是否为空。
在C语言中,使用rand()函数生成随机数时,需要先使用srand()函数设置一个种子。如果没有设置种子,每次程序运行时都会生成相同的随机数序列。例如:
for (int i = 0; i < 10; i++) {
printf("%d ", rand()); // 输出相同的数字序列
}
如果没有使用srand()函数设置种子,会导致每次程序运行时都会生成相同的随机数序列,因为rand()函数会根据当前时间生成一个初始的种子,并以此为基础生成伪随机数。如果不使用srand()函数改变种子,那么就使用了相同的种子,随机数序列也会相同。因此,通常建议在每次程序运行时都设置一个新的种子,比如使用time()函数来获取当前时间作为种子值,以保证生成的随机数序列足够随机。
解决方法:在程序中使用time()函数来获取一个随机的种子。
srand(time(NULL));
在C语言中,字符串是一个字符数组,以空字符'\0'结尾。但是,如果不小心忘记添加空字符,或者对字符串进行了越界访问,就会产生一些问题。例如:
char str[10] = "hello";
str[5] = 'w'; // 错误:没有添加空字符
printf("%s\n", str); // 输出“hellow”
在C语言中,字符串是以空字符('\0')结尾的字符数组。当声明一个字符数组时,数组长度必须比实际存储的字符数多1,以便存储最后的空字符。在这个例子中,我们声明的字符数组str的长度是10,存储了5个字符"hello"和1个空字符('\0')。当我们将第6个字符赋值为'w'时,虽然数组中确实存在了'w'字符,但是并没有相应的空字符跟随它,因此该字符数组并不是一个合法的字符串。
由于printf()函数使用空字符('\0')来确定字符串的结束位置,因此,当该字符串不包含空字符('\0')时,printf()将继续输出紧接着它内存位置后面的任何内容,直到找到空字符为止(如果根本找不到则会导致未定义的行为)。而在该示例中,恰好紧跟在字符数组str后面的内存区域存放的可能是其它的数据,因此printf()函数可能会输出一些我们不希望看到的东西。要修正这种问题,需要在修改完字符串之后手动添加一个空字符('\0')作为结尾,使得该数组成为一个正确的C风格字符串:
char str[10] = "hello";
str[5] = 'w';
str[6] = '\0';
printf("%s\n", str);
这样输出的结果就是"hello"后面跟着一个空格和"w"。
在编写循环时,如果条件不正确,就可能导致死循环或者根本没有执行循环体。例如:
int i = 0;
while (i < 10) {
printf("%d ", i);
}
这个循环中,条件i<10永远为真,所以循环将一直执行下去。
解决方法:仔细检查循环条件,确保它能够正确终止循环。
C语言中的变量有不同的作用域,如果没有理解这个概念,就容易出现一些错误。例如:
int x = 1;
if (x == 1) {
int y = 2;
}
printf("%d\n", y); // 错误:y的作用域在if语句块中
解决方法:理解变量的作用域,并确保变量在正确的位置定义和使用。
在C语言中,类型转换是一个常见的操作,但也容易引起一些错误。例如:
int a = 5;
double b = 2.0;
printf("%f\n", a / b); // 输出错误的结果
在这段代码中,a是一个整数型变量,b是一个双精度浮点数型变量。当进行除法运算时,编译器会执行隐式类型转换,将整数型变量a转换为双精度浮点数型变量,然后再进行除法运算,得到一个双精度浮点数型的结果。由于printf()函数使用%f格式说明符来输出浮点数(包括float和double类型),因此,即便结果是整数,它也将被解释为一个浮点数并以小数形式输出。
然而,在这种情况下输出的结果可能不同于预期的结果。根据C语言中的整数除法规则,两个整数相除的结果也是一个整数,小数部分将被截断。因此,在这个例子中,5/2的结果应该是2而不是2.5。因此,正确的输出格式应该是使用%f输出一个浮点数:
int a = 5;
double b = 2.0;
printf("%f\n", (double)a / b);
这里将整数型变量a强制转换为double类型,使得整数除以浮点数时不会发生隐式类型转换,得到的结果是一个双精度浮点数型的结果,可以正确地被%f格式说明符输出。
在C语言中,函数调用是一个重要的操作,但也容易出现一些问题。例如,在调用函数时,参数的类型和数量必须与函数声明中的一致,否则会产生编译错误。例如:
int add(int a, int b) {
return a + b;
}
printf("%d\n", add(1, 2, 3)); // 错误:参数数量不正确
解决方法:确保函数调用的参数类型和数量与函数声明中的一致。
在C语言中,结构体是一种自定义数据类型,由多个成员变量组成。访问结构体成员时,需要使用“.”符号。但是,如果结构体指针为空,或者结构体成员不存在,就会产生一些错误。例如:
struct Person {
char name[10];
int age;
};
struct Person *p = NULL;
printf("%s\n", p->name); // 错误:访问空指针
解决方法:在使用结构体指针和结构体成员时,先检查它们是否为空或存在。
在C语言中,文件操作是一种重要的操作。但是,如果没有正确地打开、关闭文件,就会产生一些问题。例如:
FILE *fp = fopen("test.txt", "r");
// 操作文件...
fclose(fp);
在上面的代码中,如果fopen()函数失败,就会返回NULL指针,此时使用fclose()函数就会产生错误。
解决方法:在使用文件操作函数时,确保正确地打开、关闭文件,并检查它们的返回值。
在C语言中,宏定义是一种预处理指令,可以用来定义常量、函数等。但是,如果没有正确地使用宏定义,就可能导致程序出错。例如:
#define SQUARE(x) x * x
int a = 2;
int b = SQUARE(a + 1); // 错误:得到错误的结果
这个例子中,SQUARE(a+1)展开后变成a+1*a+1,得到了错误的结果。
解决方法:使用括号来明确宏定义中的运算顺序,并避免在宏定义中使用带有副作用的表达式。
在C语言中,多线程编程是一种复杂的技术。如果没有正确地使用线程同步机制,就会产生一些错误,例如数据竞争、死锁等。例如:
void *print_message(void *ptr) {
char *message = (char *) ptr;
printf("%s\n", message);
pthread_exit(NULL);
}
pthread_t t1, t2;
char *msg1 = "Thread 1";
char *msg2 = "Thread 2";
pthread_create(&t1, NULL, print_message, (void *) msg1);
pthread_create(&t2, NULL, print_message, (void *) msg2);
在这个例子中,两个线程会同时访问printf()函数,可能会导致输出结果错乱。
解决方法:使用同步机制来保证线程之间的正确协作。
下面是一道关于C语言陷阱的面试题,请读者尝试回答:
int a = 0, b = 1, c = 2, d = 3;
if (a++ && b-- || c++ && d--) {
printf("case - %d %d %d %d\n", a, b, c, d);
} else {
printf("case + %d %d %d %d\n", a, b, c, d);
}
请问上面的代码输出什么?为什么?
期待大家的回答和讨论!
转自:嵌入式讲堂
版权声明:本文来源网络,版权归原作者所有。版权问题,请联系删除。
往期推荐