char* get_str()
{
char str[] = "abcd";
return str;
}
char str[] ="abcd";定义了一个局部字符数组,是一个局部变量,返回它的地址肯定是一个已经释放了的空间的地址。
此函数返回的是内部一个局部字符数组str的地址,且函数调用完毕后此数组被销毁,所以你返回的指针也就指向一块被销毁的内存,这种写法是错误的。
char* get_str()
{
char* str = "abcd";
return str;
}
char* str ="abcd";表示先定义个字符串常量,并将其地址赋给str。
此函数返回的是字符串常量的地址,而像这种字符串都是属于全局的,保存在文字常量区,在编译的时候就已经分配了内存了,只有程序退出的时候才会被销毁,所以返回它的地址是没有问题的,但是你最好返回常量指针,因为你不能去改变字符串常量的值。
但str保存在栈里,后面的字符串在常量区,函数返回的时候,先将常量区的地址返回,然后作用域结束,释放str在栈里面的空间。
char str[] = "abcd";
char *str1 = "abcd";
str[0] = 't';//ok
*str1 = 't';//false
准确的说,上面两个“abcd"都是存储在常量区。常量区是可读不可写的。所以任何试图对常量区进行写的操作都是非法的,当然了,这也不是一定不可写的,你可以采取某种渠道改变常量区的内存属性。
那么为什么str[]= "abdc"; 可以写呢? 答案就在str[] = "abcd";会有一个额外的拷贝过程,即把常量区的 "abcd"拷贝到栈内存去,所以就可以写了。
const char str1[] = "abcd";
const char str2[] = "abcd";
const char* p1 = "abcd";
const char* p2 = "abcd";
其中str1[]和str2[]都在栈中分配内存,地址不同;p1和p2指向常量区,值都是“abcd”,所以是同一常量,地址相同。