为什么在 char** 转换后会出现分段错误?

Why do I get a segmentation fault after char** cast?

我正在尝试做这样的事情:

int main()
{
    char strMat[][32] = { {"FIRST"}, {"SECOND"}, };
    printf ("%s\n", strMat[0]);
    test ((char **) strMat);
    return 0;
}

void test (char **strMat)
{
    printf ("%s\n", strMat[0]);
}

我一直没弄清楚为什么在调用test()之前第一个字符串写得正确,但后来我得到了一个段错误。在代码的其他部分,我用 argv 调用 test() 并且它工作正常。为什么尝试在测试函数上打印 strMat[0] 会导致分段错误?

A char[2][32] 不是 char** 而是一个连续的 2*32 字节内存块。这可以通过打印指针来突出显示:

printf("%p %p %p\n", strMat[0], &strMat[0], &strMat[1]);

当您将类型强制转换为 char** 时,您会将数组中包含的数据解释为不可避免地引用无效内存的内存地址。

事实上,如果您尝试打印您认为正确的无效地址,例如:

printf("%p\n", ((char**)strMat)[0]);

你得到 0x5453524946,它被解释为像 0x54 0x53 0x52 0x49 0x46 这样的字节数组,产生 'T' 'S' 'R' 'I' 'F',它显示了问题(字符被颠倒了,因为我假设有点端平台)。