C语言定义字符串的两种方法的区别,char * s = “aaa“ 和 char s[] = “aaa“

C语言定义字符串的两种方法的区别,char * s = “aaa” 和 char s[] = “aaa”

一张表看清楚

char s[] = “aaa”char * s = “aaa”
s是数组s 是指针
sizeof(s) = 4字节sizeof(s) = 8字节
(s == &s) 为真(s == &s) 为假
字符数组存储在栈区变量s存储在栈区,但常量“aaa”存储在全局静态存储区
s = "bbb"编译不通过,数组名不是合法的左值s =“bbb” 编译通过
s++ 不合法s++ 合法
s[0] = ‘M’ 合法s[0] = ‘M’ 不合法,因为只读代码区不可修改

有趣的案例

今天在尝试局部变量作用域问题时,写了如下代码

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>

char* getStr(int flag);
int main()
{
    char* a = getStr(1); 	/*获取sa内容*/
    char* b = getStr(0);  /*获取sb内容*/
    printf("字符串a = %s\n字符串b = %s", a, b);

}
char* getStr(int flag)
{
    char sa[] = "aaa";
    char* sb = "bbb";
    if (flag) return sa;
    else return sb;
}

预期: 因为“aaa”、“bbb”都是函数 getStr()中的局部变量,因此在跳出该函数后,两者都应该销毁,所以主函数中a串和b串都应该打印错误。
结果:
在这里插入图片描述
字符串a确实乱码,说明其内容被销毁了,而b串能打印成功,且并非偶然。

原因说明

在C语言中,char sa[] = “aaa” 和 char *sb = "bbb"定义的都是字符串,但是他们在内存中的存放形式完全不同。

char sa[] = “aaa”是在栈上分配了4个字节的内存,并将字符串"aaa"的内容拷贝到这块内存,因为sa是局部变量,当getStr函数调用完毕(返回后),sa的内存区域会被回收,所以main函数中通过指针 a 访问的字符串可能是不可预期的值。

而char* sb = “bbb"是编译期间将字符串"bbb"放到了全局/静态存储区,然后sb存储的是这个全局/静态存储区的地址。这部分内存直到程序结束才会被释放,所以main函数中可以通过指针 b 正常访问到字符串"bbb”。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值