C语言文字常量区
C语言中字符串定义
C语言的基本数据类型中并没有字符串类型,在使用的过程中通过指针或字符数组来实现
char str1[] = "abcd";
char *str2 = "abcd";
两种形式的区别
- 于str1在内存中的存放方式是{'a','b','c','d','\0'},是以字符数组的形式存放在内存中,在函数定义时存放在栈区,函数结束就释放。而str2存放在字符常量区,即全局区,当程序结束才释放。
#include <stdio.h>
char *test1()
{
char *p = "abc";
return p;
}
char *test2()
{
char p[] = "abc";
return p;
}
int main(int argc, char const *argv[])
{
char *p = NULL;
char *q = NULL;
p = test1();
q = test2();
printf("%s\n%s\n", p, q);
printf("%p\n%p\n", p, q);
return 0;
}
***************************************************
(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$ ./a.out
abc
IG˳�����
0x10fc84fa6
0x7ffedff7b94c
(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$
上述代码可以看出两者分配的区域不同,在test1函数中,字符串定义在全局区,程序结束才会释放,所以通过返回值可以将地址返回给指针,指针可以指向正常的内存空间,而在test2函数中通过字符数组来定义,那么字符串是存放在栈区,那么当函数test2结束时,空间就会被释放,所放回的指针在内存中的指向没有内容,所以输出的就是乱码。而且从两个指针指向的地址可以看出两者并不在一块内存空间。
- str1可以通过str[n]下标的方式修改单个字符,而str2是存放在文字常量区,即全局区,不可以改变,通过下标修改会报错。
int main(int argc, char const *argv[])
{
char a[] = "abcd"; # 分配在栈区
char *b = "bcd"; # 分配在文字常量区。不可改变
printf("%p\n%p\n", a, b);
a[1] = 'b';
printf("%c\n", a[1]);
b[1] = 'b';
printf("%c\n", b[1]);
return 0;
}
**********************运行结果**********************************
(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$ gcc 01_文字常量区.c
(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$ ./a.out
0x7ffee3519963
0x10c6e6f96
aacd
Bus error: 10
(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$
从运行结果可以看出,a字符串中的元素被正常修改,而b 中修改就会导致err使得程序无法正常运行下去。
- 文字常量区相同字符串共用内存,可以有效防止重复内容占用内存
int main(int argc, char const *argv[])
{
char *p = "bcd";
char *q = "bcd";
char a[] = "abcd";
char b[] = "abcd";
printf("%s\n%s\n", p, q);
printf("%p\n%p\n", p, q);
printf("%s\n%s\n", a, b);
printf("%p\n%p\n", a, b);
return 0;
}
*******************运行结果******************************
(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$ gcc 01_文字常量区.c
(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$ ./a.out
bcd
bcd
0x1044a0f9a
0x1044a0f9a
abcd
abcd
0x7ffeeb75f96b
0x7ffeeb75f966
(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$
从运行结果可以看出p和q都是指向文字常量区相同的字符串,所以两者地址相同,而a和b定义在栈区,所以分配到了不同的地址,因为“abcd”的实际存储是{'a','b','c','d','\0'}占用五个字节,所以a和b刚好相差5.
原文链接:https://blog.csdn.net/wtzhu_13/article/details/105156049