我有两个文件,User.h 和 test.c:
用户.h
include <stdio.h>
#include <string.h>
struct User {
char name[21];
};
struct User newUser(char* name) {
struct User newUser;
memset(newUser.name, '\0', 21); // ensure string ends with '\0'
memcpy(newUser.name, name, 20); // copy first 20 chars of string
return newUser;
}
测试.c
#include "User.h"
int main() {
struct User testUser = newUser(34);
printf("name is: %s\n", testUser.name);
return 0;
}
我故意将错误类型的参数传递给函数,但试图避免段错误:
如果我要将任意长度的字符串传递给 newUser(),此函数最多需要 20 个字符并将其存储在 .name 中而不会出现段错误,因为它保证以空字节结尾。
这里我传递的是一个整数。因为它需要一个字符串,所以我得到了一个编译器警告,但它还是编译了。运行时,出现段错误。
我想当函数读取 name[21] 数组时会发生段错误,但如果它保证有一个空字节,为什么它会继续读取它?它期待一个字符串,难道它不应该像对待字符串一样对待任何参数并在 '\0' 处终止吗?
看来我的逻辑有问题,有人可以教育我这里到底发生了什么吗?
最佳答案
I am intentionally passing the wrong type of argument to a function, but trying to avoid a segfault.
这就像说我要下海但尽量避免被淋湿一样。
当你做一些非法的事情时,你最终只能调用undefined behavior这可能会导致段错误。
避免的最佳方法是编写正确的代码。
问题是,该函数需要一个 char*
而您传递的是一个 int
。无论如何,这是不允许的。这是错误的,您绝不能忽略编译器警告。
详细来说,该函数需要一个指向char
(char*
) 类型的指针,并且该代码涉及从指针指向的地址位置进行读取。当您将 int
传递给函数时(忽略编译器警告),代码会尝试访问由提供的整数值指向的 memory,这很可能是 < em>无效 内存位置。因此,这种访问无效内存位置的尝试会调用 UB。
关于c - 如何防止 C 中的段错误(当采用错误类型的参数时),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39296219/