char *name和char name[ ]的区别

      在编程中我们经常会遇到向程序输入一个字符串的情况。此时,我们会有两种选择:一是定义一个字符指针,并把该字符串首字符的地址传给他;二是定义一个字符数组,并把字符串存入该字符数组中。

      那么这两种方法有什么区别呢?请大家看下面这段代码:

6217760-12b3375a9d8c628f.png
图片发自简书App

大家想想,这段代码能正常运行吗?

答案是否定的!

Why?

Reason:  A variable that points to a string literal can’t be used to change the contents of the string.

      因为字符指针cards指向的是字符串常量的地址,那么既然是常量其值必定不能被改写!所以代码中所有的向cards[ ]赋值的语句都是非法的!

代码的内存分配机理如图所示:

6217760-966f6b33dc64adb7.png
图片发自简书App

1.首先编译器将代码中的字符串JQK常量存在内存中的数据段。

2.代码定义了一个字符型指针变量cards,所以编译器在堆空间中分配了一个单元给它。

3.代码将字符串JQK在数据段中的首地址赋给cards,也就是使cards指向数据段中的字符串JQK。

4.代码要求取出字符串中的第二个字符,并将其的值赋给字符串中的第三个字符单元。这里就有问题了,因为字符串所在单元是只读的,无法完成写输入。

所以,当我们将一个字符串常量赋给一个指针时指定要用const关键词来限制,即

const char *name = "leon";

那么怎样我们才能更改字符串呢?

答案想必大家也猜到了。对,就是用char name[ ].

请看下面的代码:

6217760-a4d328884102f4d5.png
图片发自简书App

      用一个字符数组来保存我们输入的字符串!代码的内存分配机理如图所示:

6217760-00347e4ffcf1ba59.png
图片发自简书App

1.首先编译器将代码中的字符串JQK常量存在内存中的数据段。

2.代码定义了一个大小未定义的字符数组cards,编译器在堆空间中分配了字符串长度+1个单元给它,然后从数据段将字符串copy到数组中并在数组最后一个单元存入'\0',以作为字符串结束标志。

3.而后对cards数组中的操作由于是在堆空间,所以具有读写权限。

程序的内存分布:

代码通过编译器编译之后形成可在内存中实际运行的进程,它的数据在内存中的分布又是怎样的呢?请看下图:

6217760-f002fdf5f9cb19b3.png
图片发自简书App

在程序所占内存空间,从低地址向高地址方向来看,

最先存储的是程序代码数据,里面保存着组成程序的各条代码;

而后是程序中的常量数据(例如字符串常量),以上两块的读写权限是只读的,一旦写入就不可以改变了;

接着是全局变量(在源文件中,且定义在函数体外的变量),它们可以在程序运行期间随时访问和修改,且值保持不变;

再是函数栈空间,用于分配程序在运行期间申请的内存空间,即由malloc系列函数动态申请的空间。

最后是函数堆空间,用于存储程序的本地变量,即函数体内定义的自动变量。他们的增长方向是朝着低地址空间的,即最先分配的空间拥有最好地址。他们的生命周期是函数调用期,其值随着函数的调用结束而消失。

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Leon_George

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值