踩坑了!嵌入式C语言常见的几个陷阱!你遇到过吗?

小麦大叔 2025-01-10 12:03

点击上方,选择“置顶/星标公众号”

福利干货,第一时间送达

C语言是一种非常流行的编程语言,因为它简单易学,且广泛应用于各个领域。但是,由于C语言本身的特性,它也容易引起一些错误和陷阱,这些错误可能导致程序崩溃、数据丢失或者安全漏洞等问题。本文将介绍15个常见的C语言陷阱,并给出相应的解决方法。

结尾有一道面试题,欢迎大家讨论学习

1. 运算符优先级

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--先被执行,然后才是乘法运算。

解决方法:正确理解各个运算符的优先级,并使用括号来明确表达式中各个部分的计算顺序。

2. 大小写敏感

在C语言中,变量名和函数名是大小写敏感的。也就是说,myVar和MyVar是两个不同的变量名。这很容易引起混淆和错误,例如:

int MyVar = 5;
int myvar = 3;
printf("%d\n", MyVar + myvar); // 输出8

解决方法:保持一致性,使用统一的命名规则来避免混淆。

3. 数组越界

数组越界是指访问数组时超出了数组边界的范围。这种情况可能导致程序崩溃或者数据被破坏。例如:

int arr[3] = {1, 2, 3};
int x = arr[3]; // 访问越界

解决方法:注意数组的边界范围,避免访问超出范围的元素。

4. 整型溢出

在C语言中,整型溢出是一个常见的问题。当一个整数超出了它所能表示的范围时,它的值会发生“环绕”,即从最大值变成最小值,或者从最小值变成最大值。例如:

unsigned char x = 255;
x += 1; // 此时x的值为0

解决方法:使用合适的数据类型,避免超出它们所能表示的范围。

5. 指针问题

指针是C语言中的一个重要概念,但也容易引起一些错误。例如,当一个指针被赋值为NULL后,如果没有判断就继续使用它,就会产生一些奇怪的结果:

int *p = NULL;
*p = 5; // 错误:访问空指针

解决方法:在使用指针之前,检查它是否为空。

6. 随机数种子

在C语言中,使用rand()函数生成随机数时,需要先使用srand()函数设置一个种子。如果没有设置种子,每次程序运行时都会生成相同的随机数序列。例如:

for (int i = 0; i < 10; i++) {
printf("%d ", rand()); // 输出相同的数字序列
}

如果没有使用srand()函数设置种子,会导致每次程序运行时都会生成相同的随机数序列,因为rand()函数会根据当前时间生成一个初始的种子,并以此为基础生成伪随机数。如果不使用srand()函数改变种子,那么就使用了相同的种子,随机数序列也会相同。因此,通常建议在每次程序运行时都设置一个新的种子,比如使用time()函数来获取当前时间作为种子值,以保证生成的随机数序列足够随机。

解决方法:在程序中使用time()函数来获取一个随机的种子。

srand(time(NULL));

7. 字符串处理

在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"。

8. 循环条件

在编写循环时,如果条件不正确,就可能导致死循环或者根本没有执行循环体。例如:

int i = 0;
while (i < 10) {
printf("%d ", i);
}

这个循环中,条件i<10永远为真,所以循环将一直执行下去。

解决方法:仔细检查循环条件,确保它能够正确终止循环。

9. 变量作用域

C语言中的变量有不同的作用域,如果没有理解这个概念,就容易出现一些错误。例如:

int x = 1;
if (x == 1) {
int y = 2;
}
printf("%d\n", y); // 错误:y的作用域在if语句块中

解决方法:理解变量的作用域,并确保变量在正确的位置定义和使用。

10. 类型转换

在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格式说明符输出。

11. 函数调用

在C语言中,函数调用是一个重要的操作,但也容易出现一些问题。例如,在调用函数时,参数的类型和数量必须与函数声明中的一致,否则会产生编译错误。例如:

int add(int a, int b) {
return a + b;
}
printf("%d\n", add(1, 2, 3)); // 错误:参数数量不正确

解决方法:确保函数调用的参数类型和数量与函数声明中的一致。

12. 结构体访问

在C语言中,结构体是一种自定义数据类型,由多个成员变量组成。访问结构体成员时,需要使用“.”符号。但是,如果结构体指针为空,或者结构体成员不存在,就会产生一些错误。例如:

struct Person {
char name[10];
int age;
};
struct Person *p = NULL;
printf("%s\n", p->name); // 错误:访问空指针

解决方法:在使用结构体指针和结构体成员时,先检查它们是否为空或存在。

13. 文件操作

在C语言中,文件操作是一种重要的操作。但是,如果没有正确地打开、关闭文件,就会产生一些问题。例如:

FILE *fp = fopen("test.txt", "r");
// 操作文件...
fclose(fp);

在上面的代码中,如果fopen()函数失败,就会返回NULL指针,此时使用fclose()函数就会产生错误。

解决方法:在使用文件操作函数时,确保正确地打开、关闭文件,并检查它们的返回值。

14. 宏定义

在C语言中,宏定义是一种预处理指令,可以用来定义常量、函数等。但是,如果没有正确地使用宏定义,就可能导致程序出错。例如:

#define SQUARE(x) x * x
int a = 2;
int b = SQUARE(a + 1); // 错误:得到错误的结果

这个例子中,SQUARE(a+1)展开后变成a+1*a+1,得到了错误的结果。

解决方法:使用括号来明确宏定义中的运算顺序,并避免在宏定义中使用带有副作用的表达式。

15. 多线程

在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);
}

请问上面的代码输出什么?为什么?

期待大家的回答和讨论!

转自:嵌入式讲堂

版权声明:本文来源网络,版权归原作者所有。版权问题,请联系删除。



往期推荐



嵌入式开发又遇到BUG?怎么办?教你几招直接起飞

新来的同事,巧用设计模式到嵌入式软件中,这波操作秀到我了

工作十年,居然还没有掌握,“光耦” 这个电子元器件,真的太多知识点啦!今天来一起学习一下

工程师做项目,版本号怎么命名?这样做看起来很牛B!


小麦大叔 一位热衷技术的攻城狮,懂点技术,会讲故事,交个朋友?
评论
  • HDMI 2.2 规格将至,开启视听新境界2025年1月6日,HDMI Forum, Inc. 宣布即将发布HDMI规范2.2版本。新HDMI规范为规模庞大的 HDMI 生态系统带来更多选择,为创建、分发和体验理想的终端用户效果提供更先进的解决方案。新技术为电视、电影和游戏工作室等内容制作商在当前和未来提供更高质量的选择,同时实现多种分发平台。96Gbps的更高带宽和新一代 HDMI 固定比率速率传输(Fixed Rate Link)技术为各种设备应用提供更优质的音频和视频。终端用户显示器能以最
    百佳泰测试实验室 2025-01-09 17:33 112浏览
  • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球中空长航时无人机产值达到9009百万美元,2024-2030年期间年复合增长率CAGR为8.0%。 环洋市场咨询机构出版了的【全球中空长航时无人机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球中空长航时无人机总体规模,包括产量、产值、消费量、主要生产地区、主要生产商及市场份额,同时分析中空长航时无人机市场主要驱动因素、阻碍因素、市场机遇、挑战、新产品发布等。报告从中空长航时
    GIRtina 2025-01-09 10:35 99浏览
  • 故障现象一辆2017款东风风神AX7车,搭载DFMA14T发动机,累计行驶里程约为13.7万km。该车冷起动后怠速运转正常,热机后怠速运转不稳,组合仪表上的发动机转速表指针上下轻微抖动。 故障诊断 用故障检测仪检测,发动机控制单元中无故障代码存储;读取发动机数据流,发现进气歧管绝对压力波动明显,有时能达到69 kPa,明显偏高,推断可能的原因有:进气系统漏气;进气歧管绝对压力传感器信号失真;发动机机械故障。首先从节气门处打烟雾,没有发现进气管周围有漏气的地方;接着拔下进气管上的两个真空
    虹科Pico汽车示波器 2025-01-08 16:51 113浏览
  • 在过去十年中,自动驾驶和高级驾驶辅助系统(AD/ADAS)软件与硬件的快速发展对多传感器数据采集的设计需求提出了更高的要求。然而,目前仍缺乏能够高质量集成多传感器数据采集的解决方案。康谋ADTF正是应运而生,它提供了一个广受认可和广泛引用的软件框架,包含模块化的标准化应用程序和工具,旨在为ADAS功能的开发提供一站式体验。一、ADTF的关键之处!无论是奥迪、大众、宝马还是梅赛德斯-奔驰:他们都依赖我们不断发展的ADTF来开发智能驾驶辅助解决方案,直至实现自动驾驶的目标。从新功能的最初构思到批量生
    康谋 2025-01-09 10:04 97浏览
  • 光伏逆变器是一种高效的能量转换设备,它能够将光伏太阳能板(PV)产生的不稳定的直流电压转换成与市电频率同步的交流电。这种转换后的电能不仅可以回馈至商用输电网络,还能供独立电网系统使用。光伏逆变器在商业光伏储能电站和家庭独立储能系统等应用领域中得到了广泛的应用。光耦合器,以其高速信号传输、出色的共模抑制比以及单向信号传输和光电隔离的特性,在光伏逆变器中扮演着至关重要的角色。它确保了系统的安全隔离、干扰的有效隔离以及通信信号的精准传输。光耦合器的使用不仅提高了系统的稳定性和安全性,而且由于其低功耗的
    晶台光耦 2025-01-09 09:58 81浏览
  • 一个真正的质量工程师(QE)必须将一件产品设计的“意图”与系统的可制造性、可服务性以及资源在现实中实现设计和产品的能力结合起来。所以,可以说,这确实是一种工程学科。我们常开玩笑说,质量工程师是工程领域里的「侦探」、「警察」或「律师」,守护神是"墨菲”,信奉的哲学就是「墨菲定律」。(注:墨菲定律是一种启发性原则,常被表述为:任何可能出错的事情最终都会出错。)做质量工程师的,有时会不受欢迎,也会被忽视,甚至可能遭遇主动或被动的阻碍,而一旦出了问题,责任往往就落在质量工程师的头上。虽然质量工程师并不负
    优思学院 2025-01-09 11:48 114浏览
  • 在当前人工智能(AI)与物联网(IoT)的快速发展趋势下,各行各业的数字转型与自动化进程正以惊人的速度持续进行。如今企业在设计与营运技术系统时所面临的挑战不仅是技术本身,更包含硬件设施、第三方软件及配件等复杂的外部因素。然而这些系统往往讲究更精密的设计与高稳定性,哪怕是任何一个小小的问题,都可能对整体业务运作造成严重影响。 POS应用环境与客户需求以本次分享的客户个案为例,该客户是一家全球领先的信息技术服务与数字解决方案提供商,遭遇到一个由他们所开发的POS机(Point of Sal
    百佳泰测试实验室 2025-01-09 17:35 105浏览
  • 在智能网联汽车中,各种通信技术如2G/3G/4G/5G、GNSS(全球导航卫星系统)、V2X(车联网通信)等在行业内被广泛使用。这些技术让汽车能够实现紧急呼叫、在线娱乐、导航等多种功能。EMC测试就是为了确保在复杂电磁环境下,汽车的通信系统仍然可以正常工作,保护驾乘者的安全。参考《QCT-基于LTE-V2X直连通信的车载信息交互系统技术要求及试验方法-1》标准10.5电磁兼容试验方法,下面将会从整车功能层面为大家解读V2X整车电磁兼容试验的过程。测试过程揭秘1. 设备准备为了进行电磁兼容试验,技
    北汇信息 2025-01-09 11:24 97浏览
  •  在全球能源结构加速向清洁、可再生方向转型的今天,风力发电作为一种绿色能源,已成为各国新能源发展的重要组成部分。然而,风力发电系统在复杂的环境中长时间运行,对系统的安全性、稳定性和抗干扰能力提出了极高要求。光耦(光电耦合器)作为一种电气隔离与信号传输器件,凭借其优秀的隔离保护性能和信号传输能力,已成为风力发电系统中不可或缺的关键组件。 风力发电系统对隔离与控制的需求风力发电系统中,包括发电机、变流器、变压器和控制系统等多个部分,通常工作在高压、大功率的环境中。光耦在这里扮演了
    晶台光耦 2025-01-08 16:03 88浏览
  • 1月9日,在2025国际消费电子展览会(CES)期间,广和通发布集智能语音交互及翻译、4G/5G全球漫游、随身热点、智能娱乐、充电续航等功能于一体的AI Buddy(AI陪伴)产品及解决方案,创新AI智能终端新品类。AI Buddy是一款信用卡尺寸的掌中轻薄智能设备,为用户带来实时翻译、个性化AI语音交互助手、AI影像识别、多模型账户服务、漫游资费服务、快速入网注册等高品质体验。为丰富用户视觉、听觉的智能化体验,AI Buddy通过蓝牙、Wi-Fi可配套OWS耳机、智能眼镜、智能音箱、智能手环遥
    物吾悟小通 2025-01-09 18:21 18浏览
  • 1月7日-10日,2025年国际消费电子产品展览会(CES 2025)盛大举行,广和通发布Fibocom AI Stack,赋智千行百业端侧应用。Fibocom AI Stack提供集高性能模组、AI工具链、高性能推理引擎、海量模型、支持与服务一体化的端侧AI解决方案,帮助智能设备快速实现AI能力商用。为适应不同端侧场景的应用,AI Stack具备海量端侧AI模型及行业端侧模型,基于不同等级算力的芯片平台或模组,Fibocom AI Stack可将TensorFlow、PyTorch、ONNX、
    物吾悟小通 2025-01-08 18:17 84浏览
  • 职场是人生的重要战场,既是谋生之地,也是实现个人价值的平台。然而,有些思维方式却会悄无声息地拖住你的后腿,让你原地踏步甚至退步。今天,我们就来聊聊职场中最忌讳的五种思维方式,看看自己有没有中招。1. 固步自封的思维在职场中,最可怕的事情莫过于自满于现状,拒绝学习和改变。世界在不断变化,行业的趋势、技术的革新都在要求我们与时俱进。如果你总觉得自己的方法最优,或者害怕尝试新事物,那就很容易被淘汰。与其等待机会找上门,不如主动出击,保持学习和探索的心态。加入优思学院,可以帮助你快速提升自己,与行业前沿
    优思学院 2025-01-09 15:48 102浏览
  • 车机导航有看没有懂?智能汽车语系在地化不可轻忽!随着智能汽车市场全球化的蓬勃发展,近年来不同国家地区的「Automotive Localization」(汽车在地化)布局成为兵家必争之地,同时也是车厂在各国当地市场非常关键的营销利器。汽车在地化过程中举足轻重的「汽车语系在地化」,则是透过智能汽车产品文字与服务内容的设计订制,以对应不同国家地区用户的使用习惯偏好,除了让当地车主更能清楚理解车辆功能,也能进一步提高品牌满意度。客户问题与难处某车厂客户预计在台湾市场推出新一代车款,却由于车机导航开发人
    百佳泰测试实验室 2025-01-09 17:47 16浏览
  • Snyk 是一家为开发人员提供安全平台的公司,致力于协助他们构建安全的应用程序,并为安全团队提供应对数字世界挑战的工具。以下为 Snyk 如何通过 CircleCI 实现其“交付”使命的案例分析。一、Snyk 的挑战随着客户对安全工具需求的不断增长,Snyk 的开发团队面临多重挑战:加速交付的需求:Snyk 的核心目标是为开发者提供更快、更可靠的安全解决方案,但他们的现有 CI/CD 工具(TravisCI)运行缓慢,无法满足快速开发和部署的要求。扩展能力不足:随着团队规模和代码库的不断扩大,S
    艾体宝IT 2025-01-10 15:52 38浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦