没有定义未使用数组的分段错误

Segmentation fault without defining an unused array

我正在尝试编写一个简单的程序,以 16 位块的形式以十六进制输出二进制文件 (Game Boy ROM) 的前 16 KB。然而,在 for 循环期间,我的程序总是会发生段错误,但它总是在数组中的不同点发生段错误。这是代码:

#include <stdio.h>
#include <stdint.h>

int main ()
{
    uint16_t buffer[8000];
    FILE* ROM = fopen("rom.gb", "rb");
    if (ROM == NULL)
    {
        printf("Error");
        fclose(ROM);
        return 1;
    }
    fread(buffer, sizeof(buffer), 1, ROM);
    int i;
    for(i = 0; i < sizeof(buffer); ++i)
    {
        if (buffer[i] < 16)
        {
            printf("000%x ", buffer[i]);
        }
        else if (buffer[i] < 256)
        {
            printf("00%x ", buffer[i]);
        }
        else if (buffer[i] < 4096)
        {
            printf("0%x ", buffer[i]);
        }
        else
        {
            printf("%x ", buffer[i]);
        }
    }
    fclose(ROM);
    return 0; 
}

在我改为使用 uint16_t 而不是 char 之前(因为 Game Boy 有 16 位地址空间),这并没有发生,事实上,如果我包含声明

unsigned char buffer2[16000]; 

在第一个缓冲区的声明旁边,我得到了预期的输出。所以我的问题是,为什么添加一个未使用的变量会阻止程序出现段错误?我怎样才能避免必须这样做并声明一个在程序中完全未使用的巨大数组?

这一行:

for(i = 0; i < sizeof(buffer); ++i)

sizeof(buffer) 是以字节为单位的数组大小,如果你想要元素的数量使用

i < (sizeof(buffer) / sizeof(buffer[0]))

我不确定你是否正确使用了 fread,如果你查看 documentation,第二个参数是每个元素的大小,而你传递的是整个缓冲区。

您可能想交换第二个和第三个参数,请参阅 here 了解用法示例。

另请注意,正如 Alter Mann 所指出的,您需要将缓冲区的大小除以其类型的大小。