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 abcIG˳�����0x10fc84fa60x7ffedff7b94c(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 0x7ffee35199630x10c6e6f96aacdBus 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 bcdbcd0x1044a0f9a0x1044a0f9aabcdabcd0x7ffeeb75f96b0x7ffeeb75f966(base) BigfishdeMacBook-Pro:职工管理系统 bigfish$
从运行结果可以看出p和q都是指向文字常量区相同的字符串,所以两者地址相同,而a和b定义在栈区,所以分配到了不同的地址,因为“abcd”的实际存储是{‘a’,‘b’,‘c’,‘d’,’\0’}占用五个字节,所以a和b刚好相差5.