最近在Qt中经常碰到段错误,查阅网上资料得知,段错误是指程序访问(读写)了系统未给予读写权限的内存空间。
段错误出现的可能原因
段错误主要有以下几种可能:
访问了不存在的内存空间;访问了系统保护的空间;对只读内存空间写覆盖等。常见的形式有数组越界访问;野指针操作等。
一旦一个程序发生了越界访问,cpu就会产生相应的保护,于是segmentation fault就出现了。段错误应该就是访问了不可访问的内存,这个内存区要么是不存在的,要么是受到系统保护的,还有可能是缺少文件或者文件损坏。
典型的段错误
第一个中,"hello world" 是字符串常量。对于 char *s = "hello world";
把"hello world"的值----也就是字符串常量字面值,也就是"hello world"的地址,准确来说是起始地址----赋给字符指针 s,Linux下,"hello world"字符串常量是存放于只读数据区的,一般来说,32位机器上,在Linux中,堆,全局数据,常量等都是存放于从0x8048000开始的内存地址,向上增长。可以打印一下"hello world"的地址来进行验证。
char *s = "hello world",就是把"hello world"的首地址赋给s,所以s 存放的是一个只读数据区的地址,对只读区的数据进行写操作是禁止,具体由相应的操作系统进行判断以及处理。
有一次是,创建了一个 Rawdata *myRawdata[100]; 指针数组,进行了声明,却忘记初始化指针指向的空间,从而造成了段错误,
在使用前,必须先初始化指针指向的空间:
for(int i =0;i<100;i++){
myRawdata[i] = new Rawdata;
}
这样就解决了问题。
还有一次是打印数组内容时,将index++放在了打印之前,造成打印的数组空间还是空的,出现了段错误。
出现段错误如何排查呢
我是使用Qt中的debug功能,首先开始调试,然后提示出现段错误
查看堆栈视图(这个一般看最上面的函数),发现错误在widget.cpp中的第30行
接着将myRawData_index添加到表达式求值器中,发现myRawData_index的值为0,说明还没开始赋值就发生了错误,判断是未进行初始化操作,即数组指针虽然已经有了内存空间,但是指针指向的RawData并未创建内存空间。