指针指向的字符串不能修改
2013-02-21 17:30
225 查看
char* str = "resource";
str[6] = 'k'; //这句报内存写入错误
*(str+6) = 'k'; //这样写同样报错
但是这样就没问题:
char* str = new char[12];
strcpy(str, "resource");
str[6] = k; //没问题
*(str+6) = 'k'; //没问题
这样也没问题:
char str[] = "resource";
str[6] = 'k';
*(str+6) = 'k';
从网上搜寻资料后,找到了一个比较准确的解答(斜体为网上内容,来源:http://bbs.chinaunix.net/viewthread.php?tid=1084610&page=1#pid8256820):
"resource"是字符串常量。
对于 char *str = "resource";
把"resource"的值----也就是字符串常量字面值,也就是"resource"的地址,准确来说是起始地址----赋给字符指针 str,Linux下,"resource"字符串常量是存放于只读数据区的,一般来说,32位机器上,在Linux中,堆,全局数据,常量等都是存放于从0x8048000开始的内存地址,向上增长。可以打印一下"resource"的地址来进行验证。char *str = "resource",就是把"resource"的首地址赋给str,所以str 存放的是一个只读数据区的地址,对只读区的数据进行写操作是禁止,具体由相应的操作系统进行判断以及处理。
而对于 char str[] = "resource";
str[]是一个字符数组,编译器首先在栈中分配一定的连续空间用于存放“resource”中的字符以及结尾符,然后把字符串常量的内容,也就是
"resource"中的各个字符和结尾符复制到这个栈中的连续空间中。str是数组名,用来表示这个连续栈空间的起始地址,所以str中存放的是栈地址,这个地址的数据是可写的。一般来说,32位机器上,在Linux中,栈地址空间从3G(0xbfffffff)开始向下增长。
可以用语句printf("%x/n", str)来打印出str中存放的地址,来验证一下这个地址属于栈还是属于只读数据区。
而对于char* str = new char[12];
由于重新分配了内存,然后将常量字符串内容拷贝过来,这样str指向的不是只读数据区,也就可以修改了。
但是要注意的是,下面的代码也是不行的:
或许通过我的这个例子大家可以更容易的理解这个问题,在编译器调试状态下,可以很容易地看到,刚分配完内存的p的地址与给p赋值后的地址明显不同,所以经过后一个赋值语句后,p又重新指向了只读内存区,所以又不能修改了!
str[6] = 'k'; //这句报内存写入错误
*(str+6) = 'k'; //这样写同样报错
但是这样就没问题:
char* str = new char[12];
strcpy(str, "resource");
str[6] = k; //没问题
*(str+6) = 'k'; //没问题
这样也没问题:
char str[] = "resource";
str[6] = 'k';
*(str+6) = 'k';
从网上搜寻资料后,找到了一个比较准确的解答(斜体为网上内容,来源:http://bbs.chinaunix.net/viewthread.php?tid=1084610&page=1#pid8256820):
"resource"是字符串常量。
对于 char *str = "resource";
把"resource"的值----也就是字符串常量字面值,也就是"resource"的地址,准确来说是起始地址----赋给字符指针 str,Linux下,"resource"字符串常量是存放于只读数据区的,一般来说,32位机器上,在Linux中,堆,全局数据,常量等都是存放于从0x8048000开始的内存地址,向上增长。可以打印一下"resource"的地址来进行验证。char *str = "resource",就是把"resource"的首地址赋给str,所以str 存放的是一个只读数据区的地址,对只读区的数据进行写操作是禁止,具体由相应的操作系统进行判断以及处理。
而对于 char str[] = "resource";
str[]是一个字符数组,编译器首先在栈中分配一定的连续空间用于存放“resource”中的字符以及结尾符,然后把字符串常量的内容,也就是
"resource"中的各个字符和结尾符复制到这个栈中的连续空间中。str是数组名,用来表示这个连续栈空间的起始地址,所以str中存放的是栈地址,这个地址的数据是可写的。一般来说,32位机器上,在Linux中,栈地址空间从3G(0xbfffffff)开始向下增长。
可以用语句printf("%x/n", str)来打印出str中存放的地址,来验证一下这个地址属于栈还是属于只读数据区。
而对于char* str = new char[12];
由于重新分配了内存,然后将常量字符串内容拷贝过来,这样str指向的不是只读数据区,也就可以修改了。
但是要注意的是,下面的代码也是不行的:
char* p = new char[12]; p = "resource"; p[6] = 'k'; //报错
或许通过我的这个例子大家可以更容易的理解这个问题,在编译器调试状态下,可以很容易地看到,刚分配完内存的p的地址与给p赋值后的地址明显不同,所以经过后一个赋值语句后,p又重新指向了只读内存区,所以又不能修改了!
相关文章推荐
- 指向字符串的指针为何不能用来修改此字符串(c语言)http://blog.sina.com.cn/s/blog_68ff96b70101him1.html
- 为什么不能修改指向字符串的指针
- 指针为何不能修改其指向的常量字符串
- 关于字符串指针指向内容不能修改的问题
- 指向字符串的指针为何不能用来修改此字符串
- 指针到底能不能修改其指向的常量字符串?&向NULL地址copy数据
- 指针到底能不能修改其指向的常量字符串?
- 指针指向常量的场合,不能试图通过修改指针指向的内容来修改常量
- 指针到底能不能修改其指向的常量字符串?
- 指针就算指向了常量也不能修改这个常量
- 关于字符串指针不能修改的疑问
- 对字符指针指向字符串不能更改
- char * 指向的字符串不能修改
- 不能通过键盘输入一个字符串,并使未初始化的指针指向它
- char* 指向内容不能修改
- 关于字符串指针不可修改的问题
- 不能通过改变形参指针的值(形参所指向的地址)来改变实参指针的地址,实现改变实参指针所指向地址变量的值。
- C语言的字符指针必须分配空间或者指向某个地址否则不能使用
- 关于字符串指针不可修改的问题
- 编写一个函数new,对n个字符开辟连续的存储空间,此函数应返回一个指针(地址),指向字符串开始的空间。new(n)表示分配n个字节的内存空间。