空指针一定要注意

博客强调了非空判断的重要性,在信息技术中,非空判断是避免程序出现错误和异常的关键操作,能保障程序的稳定性和可靠性。

非空判断很重要

<think>我们遇到了一个段错误问题。代码中,我们声明了一个指针`items`并初始化为`NULL`,然后试图对这个空指针进行解引用(`*items = ...`)。这会导致段错误,因为不能向`NULL`指针指向的内存写入数据。 解决思路: 1. 我们可以不使用指针,直接定义一个整型变量。 2. 如果题目要求必须使用指针,那么我们需要为指针分配内存(比如定义一个整型变量,让指针指向它)或者让指针指向一个有效的地址。 这里,题目要求使用指针来解决,所以我们采用第二种方法。 修改步骤: - 定义一个整型变量,例如`count`,用于存储`sscanf`的返回值。 - 让指针`items`指向这个变量,这样我们就能通过指针访问和修改这个变量。 或者,我们可以动态分配内存,但这里情况简单,直接定义一个变量更合适。 修改后的代码: ```c #include <stdio.h> int main() { const char *data = "John Doe 25 78.5"; char name1[50], name2[50]; int count; // 定义一个整型变量存储返回值 int *items = &count; // 让指针指向这个变量 // 调用sscanf,将返回值存入count(通过指针间接赋值) *items = sscanf(data, "%49s %49s", name1, name2); // 注意:为了避免缓冲区溢出,我们指定了读取的最大字符数(这里为49,因为name数组大小为50,要留一个给结束符'\0') printf("读取了 %d 个数据项:\n", *items); printf("姓名:%s %s\n", name1, name2); printf("ret: %d\n", *items); return 0; } ``` 注意: 1. 原代码中`sscanf`的格式字符串为`"%30s%s"`,但这里我们有两个字符串,而且数组大小是50,为了安全,我们使用`%49s`来读取,防止输入过长导致溢出。但是注意,原格式字符串中第一个`%30s`限制为30个字符,第二个没有限制,这可能会导致第二个字符串溢出。因此,我们改为两个都限制为49个字符(因为数组大小为50,所以最多读取49个字符,最后一个字符为结束符'\0')。 2. 原示例数据是"John Doe 25 78.5",但我们只读取两个字符串,所以会读取"John"和"Doe",而忽略后面的数字。因此,`sscanf`的返回值应该是2。 但是,原格式字符串是`"%30s%s"`,中间没有空格,这意味着它会将第一个字符串读取到空白符(括空格),然后第二个字符串紧接着读取(会跳过空白符)直到下一个空白符。所以实际上,第一个`%30s`会读取"John",然后第二个`%s`会读取"Doe"。 我们的修改中,格式字符串改为`"%49s %49s"`,中间加了一个空格,这样可以跳过两个字符串之间的空白(括多个空格、制表符等)。这样也能正确读取"John"和"Doe"。 如果原意是想将整个"John Doe"作为一个字符串读取?显然不是,因为有两个数组。所以这样修改是合理的。 运行结果: 读取了 2 个数据项: 姓名:John Doe ret: 2 这样,我们就通过让指针指向一个有效的整型变量,避免了空指针解引用导致的段错误。 另外,如果题目要求必须动态分配内存,我们可以这样做(但通常没有必要): ```c int *items = malloc(sizeof(int)); // 动态分配 if (items == NULL) { // 处理分配失败 return 1; } *items = sscanf(...); // ... 使用 ... free(items); // 最后释放 ``` 但在这个简单例子中,第一种方法(指向栈上的变量)更简单,且不需要释放。 总结:段错误的原因是空指针解引用,我们通过让指针指向一个有效的内存地址来解决了问题。 </think>你的代码存在严重的指针使用错误:**`items` 被初始化为 `NULL`,然后试图解引用这个空指针(`*items = ...`)**,这必然会导致段错误。以下是修改后的安全实现: ```c #include <stdio.h> int main() { const char *data = "John Doe 25 78.5"; char name1[50], name2[50]; // 修复指针错误:使用栈变量代替空指针 int items; // 改为普通变量存储返回值 items = sscanf(data, "%49s%49s", name1, name2); // 添加长度限制防溢出 printf("读取了 %d 个数据项:\n", items); printf("姓名:%s %s\n", name1, name2); printf("ret: %d\n", items); return 0; } ``` ### 🔍 关键修复点说明 1. **空指针解引用问题** ```c int *items = NULL; // 错误:空指针 *items = ...; // 段错误:试图写入无效地址 ``` ✅ **修复**:改用栈变量 `int items`,避免指针操作 [^1] 2. **缓冲区溢出风险** ```c sscanf(data, "%30s%s", ...) // 错误:第二个%s无长度限制 ``` ✅ **修复**:为所有字符串添加长度限制 `%49s`(数组大小50-1) 3. **输入格式优化** 原格式 `"%30s%s"` 会导致: - 输入 "John Doe" → "John" 存入 `name1` - "Doe" 存入 `name2`(空格作为分隔符) ### ️ 指针使用的核心原则 1. **禁止解引用未初始化的指针** ```c int *p; *p = 10; // 危险:随机地址写入 ``` 2. **动态内存需配对分配/释放** ```c // 正确示例 int *items = malloc(sizeof(int)); *items = sscanf(...); // ...使用后... free(items); ``` 3. **优先使用栈变量** ```c int items; // 自动分配/释放,安全高效 ``` ### 💡 扩展应用:结构体指针解析 结合引用[3]的结构体概念,更安全的指针用法: ```c struct Record { char name1[50]; char name2[50]; }; int main() { struct Record rec; struct Record *prec = &rec; // 有效指针 int count = sscanf(data, "%49s%49s", prec->name1, prec->name2); // 安全使用指针访问结构体 } ``` ### 程序输出示例 ``` 读取了 2 个数据项: 姓名:John Doe ret: 2 ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值