strtok在C中的局部变量上

tic*_*cky -1 c strtok

我在C中调用一个外部方法时遇到了一个有趣的问题,该方法尝试strtok通过引用传递给它的本地(主)字符串.如果我在main中strtok字符串,它按预期工作,但如果我调用外部方法,它会失败并出现segfault.Valgrind在访问strtok结果的第一个元素时的输出:

Address 0xfffffffffeffebc0 is not stack'd, malloc'd or (recently) free'd
Run Code Online (Sandbox Code Playgroud)

--- test.c ---

extern void tt(char* x);

main(argc, argv) 
    int argc;
    char *argv[];
{
    char line[5000];
    // if I uncomment the following two lines and comment above it works as expected
    // char * line;
    // line = malloc(5000);
    strcpy(line, "hello world");
    printf("%d\n", sizeof(line));
    tt(line);
}
Run Code Online (Sandbox Code Playgroud)

--- test2.c ---

void tt(char* l) {
    char* x = strtok(l, " \t");
    printf("%p\n", x);
    printf("%c\n", x[0]);
}
Run Code Online (Sandbox Code Playgroud)

编译

gcc -c test2.c -o test2.o
gcc test.c test2.o
Run Code Online (Sandbox Code Playgroud)

恕我直言,它应该打印出类似的东西:

./a.out
0x.....
h
Run Code Online (Sandbox Code Playgroud)

但相反,我在打印"h"时遇到了段错误.

有人可以解释一下这种行为吗?

谢谢.

Dan*_*her 7

在文件中

--- test2.c ---

void tt(char* l) {
    char* x = strtok(l, " \t");
    printf("%p\n", x);
    printf("%c\n", x[0]);
}
Run Code Online (Sandbox Code Playgroud)

你没有包含string.h,所以编译器假定返回类型为intfor strtok.这样就行了

char *x = strtok(l, " \t");
Run Code Online (Sandbox Code Playgroud)

没有指定正确的指针x(正如valgrind输出所示,你的指针至少是64位,但int很可能只有32位).然后解除引用损坏的指针会导致分段错误.

  • 这是编译警告并注意它们的原因之一.当我不包含`string.h`时,gcc会给出一个关于从没有强制转换的整数初始化指针的警告,当我添加`-Wall`时,我得到一个警告隐式声明`strtok`.如果您的编译器没有提供这些诊断,那么获得更好的:) (3认同)