C语言「NULL 真假分身」:90% 写错的空指针陷阱

简介: 在C语言中,`NULL`与`0`本质不同:前者是空指针常量(如`(void*)0`),后者是整数。变参函数中混用会导致崩溃;跨平台时`NULL`赋整型可能截断。安全规范:只含`<stddef.h>`后使用,指针判空用`p == NULL`,禁赋整型。

很多人以为 NULL == 0,但在 C 里,它们根本不是一回事

1. 两个完全不同的东西

  • 空指针常量:源码层面的 0(void *)0
  • 空指针:运行时、操作系统判定为无效地址的指针值

编译器会自动把空指针常量转为对应平台的空指针,但类型不一样

2. 致命暗坑:变参函数传 NULL

printf("%s", 0);   // 崩溃:传 int 0
printf("%s", NULL);// 正常:传指针空值

可变参数不做隐式转换,0 是整型,NULL 是指针,栈布局错位直接炸。

3. 隐蔽跨平台坑

32 位:指针 4 字节
64 位:指针 8 字节
直接写 int x = NULL;,64 位会截断高 4 字节,埋下随机崩溃。

极简安全规范

  1. 只包含 <stddef.h> 再用 NULL
  2. 指针判空只用:if (p == NULL) / if (!p)
  3. 永远不要把 NULL 赋值给整型变量
相关文章
|
3月前
|
网络协议 编译器 C语言
C语言深度解析:内存对齐与结构体填充的底层逻辑
C语言中,内存对齐是CPU硬件强制要求的底层规则,直接影响结构体大小、访问性能与硬件兼容性。合理排列成员可减少填充、节省内存;滥用`#pragma pack`则易致崩溃或性能暴跌。嵌入式、网络协议与跨平台开发必备核心知识。(239字)
420 14
|
3月前
|
编译器 程序员 C语言
C语言深度解析:未定义行为(UB)—— 90%玄学bug的根源
C语言因极致性能与硬件控制力成为系统开发首选,但其“自由”伴生未定义行为(UB):语法合法却结果不可控,是“调试正常、上线崩溃”的元凶。UB包括数组越界、有符号溢出、空指针解引用、序列点违规、重复释放等,编译器可任意优化或崩溃。规避需严守边界、开启高警告、判空置空、拆分表达式、预检溢出。(239字)
|
3月前
|
存储 网络协议 安全
C语言「内存对齐潜规则」:结构体里看不见的填充字节
内存对齐是CPU硬件要求的数据地址约束规则:变量须存于其字节大小的整数倍地址。编译器自动插入填充字节确保对齐,导致结构体体积“膨胀”、硬件寄存器读写错位或协议异常。合理排序成员(从大到小)、慎用`packed`、明确对齐控制,是嵌入式与底层开发的关键避坑要点。(239字)
|
3月前
|
安全 编译器 C语言
C语言深度解析:严格别名规则——指针强转的隐形陷阱
C语言严格别名规则禁止不同非char指针访问同一内存,是编译器激进优化(如常量折叠)的基石。违规(如强制类型转换解析协议)会导致O2下玄学bug。安全方案:优先用memcpy(零开销)、union类型双关;避免关闭-fno-strict-aliasing。
|
3月前
|
C语言
C语言深度短文:函数调用栈与栈帧原理(极简版)
很多人写C多年,却不懂函数调用的本质——栈帧。每次调用函数,CPU在栈上开辟空间保存返回地址、参数、局部变量等,即“栈帧”;函数返回即销毁该帧。局部变量快因在栈上,递归过深致栈溢出,返回局部变量地址则成野指针。懂栈帧,才真正理解C的运行机制。(239字)
|
3月前
|
Java API
Java MethodHandle:超越反射的轻量化方法调用底层引擎
Java 7引入的MethodHandle是JVM级动态调用机制,相比反射:仅一次权限校验、强类型绑定、零装箱开销、支持方法适配与invokedynamic。性能达反射3–10倍,是Lambda、动态代理及现代框架的底层引擎。(239字)
225 6
|
3月前
|
存储 安全 算法
C语言高频错误实例对比:8段代码帮你避开90%的坑
本文精选8组典型C语言错误与正确代码对比,直击数组越界、字符串溢出、野指针、内存泄漏、有无符号混用、返回局部地址、sizeof误用、未定义行为等高频陷阱,以实例培养安全编码直觉。(239字)
|
3月前
|
存储 安全 C语言
C语言深度解析:函数指针的底层本质与避坑指南
本文深入剖析C语言函数指针的本质——函数名即代码段入口地址,厘清其与数据指针的根本差异;系统梳理回调、跳转表、中断向量、动态库等核心应用场景;重点警示签名不匹配、`void*`强转、野指针调用三大致命陷阱,并给出`typedef`封装、空值校验、边界防护等最佳实践。(239字)
602 134
|
3月前
|
缓存 编译器 程序员
C语言深度解析:restrict关键字——编译器性能优化的终极钥匙
C99的`restrict`关键字是C语言性能优化的“终极钥匙”:它向编译器承诺指针独占访问内存,彻底解决同类型指针别名问题,解锁循环向量化、寄存器缓存等激进优化。滥用致未定义行为,善用则性能飙升数倍——这才是真正高阶C程序员的必修课。(239字)
|
3月前
|
存储 C语言 内存技术
C语言深度解析:大小端字节序——多字节数据的底层存储规则
大小端指CPU对多字节数据在内存中的存放顺序:大端高字节存低地址,小端反之。x86/ARM默认小端,网络字节序统一为大端。跨平台、网络通信、二进制协议开发中必须显式处理字节序转换,否则数据解析必错。
883 138