二、字符指针和字符数组
1、字符指针和字符数组的初始化问题
void demo01() {
char *p = "hello";
char a[10] = "hello";
p[0] = 'H';
a[0] = 'H';
printf("p=%s,a=%s", p, a);
}
结果:字符指针无法修改字符串,而数组a能成功修改值为“Hello”;
原因是:
(1)执行p[0] = 'H';
时报错;
写入访问权限冲突。
因为“hello”是作为字符串常量进行保存的,所以不能进行修改;
(2)字符数组a可以修改是因为,是将"hello"复制到了数组a所在的栈空间,并没有修改字符串常量“hello”的值;
可以看下面变量在内存中的变化;
指针p的地址:
指针p指向了字符串常量hello的地址;
数组a的地址:可以看到数组a是复制的“hello”到了栈空间。
2、堆空间和栈空间使用上的区别,堆空间和栈空间申请内存的区别?
(1)在函数栈上申请内存
char* demo03() {
char a[10] = "hello";
return a;
}
void demo02() {
char* p;
p = demo03();
printf("%s\n’’", p);
}
问题:
执行时会出现返回局部变量或临时变量的地址: a
这样的错误;
原因:
我们在子函数demo03中申请了一块栈上的内存空间并返回地址,这时当demo03函数执行完后,栈空间就释放了,所以这时,demo03返回的地址中,并没有了原来的数组,所以也就无法打印数组;
当执行到demo03函数中,栈空间:
继续执行函数,当执行到demo02时,
继续执行到printf,这时再看内存,发现原来数组a的数据已经没有了;
最后输出的就是乱码;
(2)在堆空间申请内存
char* demo03() {
char* a;
a = (char*)malloc(10);
//char a[10] = "hello";
strcpy(a,"hello");
return a;
}
void demo02() {
char* p;
p = demo03();
printf("%s", p);
}
结果:
在堆空间上申请的内存,不会因为子函数结束主动释放,需要进行手动释放;free(a);
总结:在函数栈上申请的内存空间,一旦函数运行完就释放了,如果想要在子函数中申请内存,可以使用堆空间;