当前位置: 首页 > news >正文

char s[]和char *s的区别,数组和指针的,堆和栈指针的一些思考

最近在学习的时候看到一个概念,数组不等价于指针,很合理但又很难理解。

例如char s[]和char *s有什么区别,前者是数组,后者是指针,个人学习成果如下:

1.char s[]和char *s的区别

char s[]:

①数组,char s[]定义了一个字符数组

②内存分配: 内存在栈上分配。
③大小固定: 一旦定义,数组的大小就不能改变。
④初始化: 可以用字符串直接初始化。

char s[] = "hello";

⑤可变性: 可以修改数组中的元素。

s[0] = 'H';  // 合法

char *s
①指针: char *s 定义了一个指向字符的指针。
②内存分配: 可以指向栈上或堆上的内存。
③大小灵活: 指针指向的内存块大小可以动态改变(如果是堆内存)。
④初始化: 可以用字符串字面量初始化,但需注意字符串字面量自身是不可修改的。

char *s = "hello";

⑤可变性: 如果指针指向的是字符串字面量,则不能修改内容。

s[0] = 'H';  // 非法,如果 s 指向的是字符串字面量

2.如何理解数组不等价于指针

①内存分配: 数组是在栈上分配内存,而指针可以指向栈或堆上的内存。
②大小信息: 数组名包含了数组大小的信息(通过 sizeof 可以获取),而单纯的指针没有。
③可修改性: 如果指针指向的是字符串字面量,那么这部分内存是不可修改的,而数组是可修改的。
④运算限制: 指针可以进行递增和递减操作来遍历内存,而数组名实际上是一个常量指针,不能进行递增和递减操作。
⑤函数传参: 当数组作为函数参数时,它会退化为指针,但在定义和初始化时,它们是不同的。
⑥类型信息: 从类型系统的角度看,char s[] 和 char *s 是不同的类型。
由于以上差异,数组和指针在 C/C++ 中是不等价的,尽管在某些特定情况(例如作为函数参数)下,数组名会退化为指针。(一般多维数组作为函数形参传递的时候,就会退化为指针)

3.指针可以指向栈或堆上的内存

不知道大家看到《2.如何理解数组不等价于指针》中的 “①内存分配: 数组是在栈上分配内存,而指针可以指向栈或堆上的内存。”会不会很难理解,我看到的时候就会在想,什么时候指向栈,什么时候指向堆。

我们先来理解一下栈和堆

栈:

①自动分配/释放:函数内的局部变量通常在栈上分配内存,当函数调用结束后,这部分内存会自动释放。
②快速访问:访问栈上的内存一般比访问堆上的内存要快。
③有限的大小:栈的大小通常是有限的,超出大小会导致栈溢出。
例子:

void function() {int a = 10;  // 在栈上分配char *p = &a;  // p 是指针,指向栈上的内存地址
}

堆:

①手动分配/释放:通过malloc()、calloc()、new等函数在堆上分配内存,需要手动释放。
②访问速度较慢:一般来说,访问堆上的内存要比访问栈上的内存慢。
③动态大小:堆的大小可以动态地增长或缩小。
例子:

void function() {char *p = (char *)malloc(10);  // p 是指针,指向堆上的内存地址// ... 使用 p ...free(p);  // 释放 p 指向的堆内存
}

这个时候就好理解了,指针如何指栈和指向堆的内存:

指向栈上的内存:当指针指向一个局部变量时,它指向栈上的内存。

int a = 10;
int *p = &a;

指向堆上的内存:当通过malloc、calloc或new分配内存后,返回的指针指向堆上的内存。

int *p = (int *)malloc(sizeof(int) * 10);

总的来说:指针是内存地址的抽象,可以指向栈上或堆上的内存,具体取决于指针是如何初始化或赋值的。

4.指针本身在堆上还是栈上

3.指针可以指向栈或堆上的内存》知道了指针指向是堆还是栈,那指针本身是在堆上还是栈上?

指针变量本身的存储位置取决于它是如何声明和分配的。

①局部指针变量:如果指针变量是在函数内部定义的,那么它存储在栈上。

void function() {int *p;  // p 是局部变量,存储在栈上
}

②全局指针变量:如果指针变量是全局变量,那么它通常存储在全局/静态存储区。

int *p;  // p 是全局变量,存储在全局/静态存储区

③动态分配的指针变量:如果指针变量本身是通过动态内存分配函数(如 `malloc` 或 `new`)获得的,那么这个指针变量(或者说,该指针变量指向的内存块)存储在堆上。

int **pp = (int **)malloc(sizeof(int *));  // pp 是局部变量,但它指向的内存块在堆上

④成员指针变量:如果指针是某个对象的成员变量,那么它的存储位置取决于对象本身的存储位置。如果对象在栈上,指针也在栈上;如果对象在堆上,指针也在堆上。

struct MyStruct {int *p;  // p 的存储位置取决于 MyStruct 实例的存储位置
};

⑤函数参数:作为函数参数传入的指针通常也是存储在栈上的。

void function(int *p) {  // p 作为函数参数,存储在栈上// ...
}

总体来说,指针变量本身就像其他任何变量一样,它的存储位置取决于它是如何被声明和分配的。指针变量存储的是内存地址,这个地址可以指向栈上、堆上或全局/静态存储区的内存。

http://www.lryc.cn/news/171233.html

相关文章:

  • Flutter快速入门学习(二)
  • 【Phoenix】phoenix实现每个Primarykey主键保留N版本数据,CDC数据记录为Changelog格式
  • 阿里云服务器开放的一个新端口,重启防火墙,端口未启动
  • 【PHPCUSTOM】打包PHP程序为EXE
  • 药品咨询报告合集整理平台打包(一共36597份)【专题推荐】
  • 数字化管理新革命,AI数字人CEO登场引领变革!
  • FPGA/数字IC(芯海科技2022)面试题 2(解析版)
  • SpringMVC之JSON数据返回与异常处理机制---全方面讲解
  • 信息化发展53
  • Java学习笔记——字符/字符串
  • 数据结构与算法基础-(1)
  • 华为云云耀云服务器L实例评测|轻量级应用服务器对决:基于 STREAM 深度测评华为云云耀云服务器L实例的内存性能
  • Windows安装Neo4j
  • vue3开发必备核心要点
  • 针对敏感数据的安全转录服务
  • leetcode 10. 正则表达式匹配
  • Vue前端开发中的输入限制与输入规则探究
  • 自己封装 vue3+ts 组件库并且发布到 NPM
  • MySQL学习系列(6)-每天学习10个知识
  • “毛细血管”的进化:华为分销业务如何让伙伴也有“高能级”
  • 警惕!多本SCI/SSCI被剔除,9月SCI/SSCI期刊目录已更新~(附下载)
  • 一点整理
  • Vulnhub系列靶机---Deathnote: 1死亡笔记
  • 从基础到高阶:史上最小白的Attention机制详解——揭秘人工智能中的核心技术
  • 9.20金融科技(比特币)
  • 什么是内存碎片?
  • C语言堆排序
  • 【学习笔记】CF573E Bear and Bowling
  • 函数扩展之——内存函数
  • 【在线机器学习】River对流数据进行机器学习