“段错误(segment fault)”,"非法操作”该内存地址不能read/write

为什么写程序经常出现“段错误(segment fault)”,"非法操作”该内存地址不能read/write的错误信息?

    这是典型的非法指针解引用造成的错误,当指针指向一个不允许读或写的内存地址,而程序却试图利用指针来读或写该地址的时候,就会出现上述的错误信息,在linux或windows的内存布局中,有些地址是始终不能读写的,例如0地址。还有一些地址是一开始不允许读写的,应用程序必须事先请求获取这些地址的读写权,或者某些地址一开始并没有映射到实际的物理内存,应用程序必须事先请求将这些地址映射到实际的物理内存地址(commit),之后才能自由读写这片内存。

    当一个指针指向这些区域的时候,对它指向的内存进行读写就会引发错误。造成这样的最普遍的原因有2个:

    1:程序员将指针初始化为NULL,之后却没有给它一个合理的值就开始使用指针。

    2:程序员没有初始化栈上的指针,指针的值一般会是随机数,之后就直接开始使用指针。

下面我们看看具体的例子:

    1:错误的访问类型

#include<stdio.h>
#include<stdlib.h>

int main()
{
	char *s = "It's me'";
	s[1] = 'i';
	return 0;	
}

“It's me”作为一个常量字符串,在编译后会被放在.rodata(GCC),最后连接生成目标程序时.rodata节会被合并到text segment与代码段放在一起,所以它所处的区域是只读的。

2:访问不属于进程地址空间的内存

#include<stdio.h>
#include<stdlib.h>

int main()
{
	int *ptr = (int *)0xC00ffff;
	*ptr = 10; 
	return 0;	
}

我们访问了一个属于内核的地址(IA32,32bit)。内核地址是不允许访问的。

3:访问不存在的内存

最常见的情况是解引用空指针。

#include<stdio.h>
#include<stdlib.h>

int main()
{
	int *ptr = NULL;
	*ptr = 10; 
	return 0;	
}

4:栈溢出

如何避免这些呢?

1:尽量参照标准写程序

2:大量使用assert,在认为不该出现的情况,就assert进行检查

3:彻底懂得你的程序,在内存的分配和释放方面,在操作每一个指针前,你都应该清楚它所指向的内存的出处(堆,栈,全局区),并清楚内存的生存周期。



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值