先明确两个概念:(1和2是两个完全不一样的东西)

1、数组指针:如果一个指针指向了数组(该数组就是普通定义的数组),我们就称它为数组指针(Array Pointer)。(1强调的是指针)

2、指针数组:数组中每个元素都是指针。(2强调的数组)

数组(Ariraayn)是一'm系列具有i相同类型的数据的集合,每一份数据叫做一个数组元素(Element)。数组中的所有元素在内存中是连续排列的,整个数组占用的是一块内存。以int arr[] = { 9, 1, 10, 700, 93 };为例,该数组在内存中的分布如下图所示:

定义数组时,要给出数组名和数组长度,数组名可以认为是一个指针,它指向数组的第 0 个元素。在C语言中,我们将第 0 个元素的地址称为数组的首地址。以上面的数组为例,下图是 arr 的指向:

PS:数组名的本意是表示整个数组,也就是表示多份数据的集合,但在使用过程中经常会转换为指向数组第 0 个元素的指针,所以上面使用了“认为”一词,表示数组名和数组首地址并不总是等价。初学者可以暂时忽略这个细节,把数组名当做指向第 0 个元素的指针使用即可,将在这两篇博客中讲解区别:数组和指针绝不等价,数组是另外一种类型,数组到底在什么时候会转换为指针。

下面的例子演示了如何以指针的方式遍历数组元素:

#include <stdio.h>
#include <iostream>
using namespace std;int main()
{int arr[] = { 9, 1, 10, 700, 93 };//用来求数组的长度,sizeof(arr) 会获得整个数组所占用的字节数,sizeof(int) 会获得一个数组元素所占用的字节数,它们相除的结果就是数组包含的元素个数,也即数组长度int len = sizeof(arr) / sizeof(int); //求数组长度//用来求数组的长度cout << "len:"<<len << endl;cout << "arr=:" << arr << endl;printf("arr = %#X", arr);int i;for (i = 0; i<len; i++) {//arr 是数组名,指向数组的第 0 个元素,表示数组首地址, arr+i 指向数组的第 i 个元素,*(arr+i) 表示取第 i 个元素的数据,它等价于 arr[i]printf("%d  ", *(arr + i)); //*(arr+i)等价于arr[i]//arr是首地址,可以加是因为数组是连续的。//arr 是int*类型的指针,每次加 1 时它自身的值会增加 sizeof(int),加 i 时自身的值会增加 sizeof(int) * i}printf("\n");return 0;
}

结果如下:

我们也可以定义一个指向数组的指针,例如:

int arr[] = { 9, 1, 10, 700, 93 };
int *p = arr;//定义指针p。p是一个指向 int 类型数据的指针变量。在定义指针变量 p的同时对它进行初始化,并将arr的地址赋予它,此时 p就指向了arr即数组的首地址。

arr 本身就是一个指针,可以直接赋值给指针变量 p。arr 是数组第 0 个元素的地址,所以int *p = arr;也可以写作int *p = &arr[0];。也就是说,arr、p、&arr[0] 这三种写法都是等价的,它们都指向数组第 0 个元素,或者说指向数组的开头。

但是此处再次说明一下:arr 本身就是一个指针”这种表述并不准确,严格来说应该是“arr 被转换成了一个指针”。

数组指针指向的是数组中的一个具体元素,而不是整个数组,所以数组指针的类型和数组元素的类型有关,上面的例子中,p 指向的数组元素是 int 类型,所以 p 的类型必须也是int *

反过来想,p 并不知道它指向的是一个数组,p 只知道它指向的是一个整数,究竟如何使用 p 取决于程序员的编码。(在上边对p进行初始化为arr,是不是就确定他指向的是首个元素了?答:是的但是他任然是一个变量,因为我们可以随时改变他的指向。)下边将上边的例子进行更改使用数组指针来遍历数组元素。

#include <stdio.h>
#include <iostream>
using namespace std;int main() {int arr[] = { 9, 1, 10, 700, 93 };int i, *p = arr, len = sizeof(arr) / sizeof(int);//p是地址,只是一个指向int类型的指针for (i = 0; i<len; i++){printf("%d  ", *(p + i));//*(p+i)是取值}printf("\n");return 0;
}

输出结果:

数组在内存中只是数组元素的简单排列,没有开始和结束标志,在求数组的长度时不能使用sizeof(p) / sizeof(int),因为 p 只是一个指向 int 类型的指针,编译器并不知道它指向的到底是一个整数还是一系列整数(数组),所以 sizeof(p) 求得的是 p 这个指针变量本身所占用的字节数,而不是整个数组占用的字节数。

也就是说,根据数组指针不能逆推出整个数组元素的个数,以及数组从哪里开始、到哪里结束等信息。不像字符串,数组本身也没有特定的结束标志,如果不知道数组的长度,那么就无法遍历整个数组。

上节我们讲到,对指针变量进行加法和减法运算时,是根据数据类型的长度来计算的。如果一个指针变量 p 指向了数组的开头,那么 p+i 就指向数组的第 i 个元素;如果 p 指向了数组的第 n 个元素,那么 p+i 就是指向第 n+i 个元素;而不管 p 指向了数组的第几个元素,p+1 总是指向下一个元素,p-1 也总是指向上一个元素。

更改上面的代码,让 p 指向数组中的第三个元素: (注意数组下标从0开始,第三个元素就是700)

#include <stdio.h>
#include <iostream>
using namespace std;int main() {int arr[] = { 9, 1, 10, 700, 93 };int *p = &arr[3];  //也可以写作 int *p = arr + 2;cout << "*p=:" <<*p<< endl;printf("%d, %d, %d, %d, %d\n", *(p - 3), *(p - 2), *(p -1), *p, *(p + 1));return 0;
}

结果是:

引入数组指针后,我们就有两种方案来访问数组元素了,一种是使用下标,另外一种是使用指针。

第一种:

1) 使用下标

也就是采用 arr[i] 的形式访问数组元素。如果 p 是指向数组 arr 的指针,那么也可以使用 p[i] 来访问数组元素,它等价于 arr[i]。

#include <stdio.h>
#include <iostream>
using namespace std;int main() {int arr[] = { 9, 1, 10, 700, 93 };int i, *p = arr, len = sizeof(arr) / sizeof(int);//p是地址,只是一个指向int类型的指针for (i = 0; i<len; i++){printf("%d  ", *(p + i));//*(p+i)是取值//cout << "p[" << i << "]=:" << *p[i] << endl;//这样是错的cout << "p[" << i << "]=:" << p[i] << endl;//规定 地址[i] 这样的形式就是取相应的元素。}printf("\n");return 0;
}

结果输出:(上述使用到了一个规定  地址[i] 这样的形式就是取对应i的元素。数组元素的访问形式可以看做 address[offset],address 为起始地址,offset 为偏移量:c1 = str[4]表示以地址 str 为起点,向后偏移4个字符,c5 = (str+1)[5]表示以地址 str+1 为起点,向后偏移5个字符,等价于str[6])

比如:

char str[20] = {0};//str+2 表示指向第 2 个元素,(str+2)[2] 相当于 *(str+2+2),也就是取得第 4 个元素的值。

第二种:

2) 使用指针

也就是使用 *(p+i) 的形式访问数组元素。另外数组名本身也是指针,也可以使用 *(arr+i) 来访问数组元素,它等价于 *(p+i)。

例子如上。

更改上面的代码,借助自增运算符来遍历数组元素:

#include <stdio.h>
#include <iostream>
using namespace std;int main() {int arr[] = { 9, 1, 10, 700, 93 };对于数组 arr,它的类型是int [],表示这是一个拥有多少个 int 数据的集合,int i, *p = arr, len = sizeof(arr) / sizeof(int);//p是地址,只是一个指向int类型的指针;对于指针变量 p,它的类型是int *;for (i = 0; i<len; i++){printf("%d  ", *p++);//自增运算符来遍历数组元素}printf("\n");return 0;
}

输出结果:

上述代码中*p++ 应该理解为 *(p++),每次循环都会改变 p 的值(p++ 使得 p 自身的值增加),以使 p 指向下一个数组元素。该语句不能写为 *arr++,因为 arr 是常量,而 arr++ 会改变它的值,这显然是错误的。(为什么arr是常量,而p是变量,上边已经对p初始化了啊?初始化为arr,不是确定指向第一个元素吗?答:是的但是我们可以改变它的指向,所以它就是一个变量,指针变量。)

总结:

假设 p 是指向数组 arr 中第 n 个元素的指针,那么 *p++、*++p、(*p)++ 分别是什么意思呢?

(1)*p++ 等价于 *(p++),表示先取得第 n 个元素的值,再将 p 指向下一个元素,上面已经进行了详细讲解。

(2)*++p 等价于 *(++p),会先进行 ++p 运算,使得 p 的值增加,指向下一个元素,整体上相当于 *(p+1),所以会获得第 n+1 个数组元素的值。

(3)(*p)++ 就非常简单了,会先取得第 n 个元素的值,再对该元素的值加 1。假设 p 指向第 0  个元素,并且第 0 个元素的值为 8,执行完该语句后,第 0  个元素的值就会变为 9。

参考:C语言中文网

数据集合包含了多份数据,直接使用一个集合没有明确的含义,将数组名转换为指向数组的指针后,可以很容易地访问其中的任何一份数据,使用时的语义更加明确。
C语言标准规定,当数组名作为数组定义的标识符(也就是定义或声明数组时)、sizeof 或 & 的操作数时,它才表示整个数组本身,在其他的表达式中,数组名会被转换为指向第 0 个元素的指针(地址)。

C指针4:数组指针(指向数组的指针)相关推荐

  1. 指针笔记(指针数组和指向数组的指针,数组中a和a的区别等)

    指针数组和指向数组的指针 int *p[4]和int (*p)[4]有何区别? 前者是一个指针数组,数组大小为4,每一个元素都是一个指向int的指针 后者是指向int[4]类型数组的指针 以上代码若运 ...

  2. c 语言 指针 指向数组,C 指向数组的指针

    您可以先跳过本章,等了解了 C 指针的概念之后,再来学习本章的内容. 如果您对 C 语言中指针的概念有所了解,那么就可以开始本章的学习.数组名是一个指向数组中第一个元素的常量指针.因此,在下面的声明中 ...

  3. C和指针之指针数组和指向数组的指针

    1.指针数组 定义一个指针数组,该数组中每个元素是一个指针,每个指针指向哪里就需要程序中后续再定义 int *p[10]; 2.指向数组的指针 定义一个数组指针,该指针指向含10个元素的一维数组(数组 ...

  4. C语言基础——指针数组(指向数组的指针)

    数组(Array)是一系列具有相同类型的数据的集合,每一份数据叫做一个数组元素(Element).数组中的所有元素在内存中是连续排列的,整个数组占用的是一块内存.定义数组时,一定要给出数组名,数组名可 ...

  5. 20返回指针的函数与指向函数的指针

    一.返回指针的函数 指针也是C语言中的一种数据类型,因此一个函数的返回值肯定可以是指针类型的. 返回指针的函数的一般形式为:类型名 * 函数名(参数列表) 比如下面这个函数,返回一个指向char类型变 ...

  6. 数组名和指针(这里为指向数组首元素的指针)区别?

    二者均可通过增减偏移量来访问数组中的元素. 数组名不是真正意义上的指针,可以理解为常指针,所以数组名没有自增.自减等操作. 当数组名当做形参传递给调用函数后,就失去了原有特性,退化成一般指针,多了自增 ...

  7. c语言memset清空指向数组的指针_C语言中数组和指针的关系

    数组的数组名其实可以看作一个指针.看下例: 1.    int array[10]={0,1,2,3,4,5,6,7,8,9},value; 2.    value=array[0]; //也可写成: ...

  8. 指向数组的指针与指向数组首元素的指针

    我的主力博客:半亩方塘 以下原创内容,转载请注明地址,欢迎对以下内容提供不同参考意见: 指向数组的指针与指向数组首元素的指针究竟有什么区别呢?有人说,这二者不是一回事么?它们就是同一个东西啊!然而,事 ...

  9. 数组的指针、指针数组以及指向指针的指针

    考虑数组的指针的时候我们要同时考虑类型和维数这两个属性.换一句话,就是说一个数组排除在其中存储的数值,那么可以用类型和维数来位置表示他的种类. 一维数组 在c和c++中数组的指针就是数组的起始地址(也 ...

  10. C/C++指向指针的指针、指向数组的指针以及存放指针的数组

    关于指针这块,有段时间没有接触了(因为最近都是在使用java),感觉有点生疏了,昨天同学问了关于一个指针的问题,当时竟然卡住了,好吧,还是基础不够牢靠啊,突然觉得掌握了东西还是要以某种形式记录下来,光 ...

最新文章

  1. JavaScript将在企业环境中引发巨大变革
  2. Hadoop科普文—常见的45个问题解答 #183; Hadoop
  3. 【 Laravel 】日 常 整 理 记 录 分 享
  4. Python3算术运算符
  5. springboot运行原理
  6. u-tools图床便捷生成markdown图片
  7. 如何查找完全二叉树最后一层的最右边的结点
  8. 实现Kubernetes跨集群服务应用的高可用
  9. 用 intellij idea 创建一个Java web项目
  10. abp调用登录接口获取token再调用其他接口
  11. dnf服务器字体文件,DNF普通伤害字体怎么改为超时空漩涡字体_字体制作方法分享_3DM网游...
  12. Android开发入门前准备
  13. 【板栗糖GIS】Win11如何取消打印机任务
  14. 中联医疗系统服务器,中联医学影像系统(PACS)
  15. 装逼利器:QQ号转换成16进制登陆--用了这么长时间QQ竟然不知道
  16. 拍摄UFP 单一职责原则
  17. Head First Design Patterns(深入浅出设计模式)-目录
  18. 淘宝、京东、天猫商品名称数据集下载最新版本。包括中药、化学器材、摄影、动物、酒水、办公事务。
  19. pvifa怎么用计算机算,年金现值系数表【完整版】
  20. 【以太网硬件七】扰码和mBnB有什么区别和相同点?

热门文章

  1. 2021年大数据Hadoop(二):Hadoop发展简史和特性优点
  2. 2021年大数据Flink(四十五):​​​​​​扩展阅读 双流Join
  3. 2021年大数据Spark(十九):Spark Core的​​​​​​​共享变量
  4. Python:CrawlSpiders
  5. 20155308『网络对抗技术』Exp7:网络欺诈防范
  6. jQuery_第五章_jQuery事件和动画
  7. CodeForces 595A
  8. Const 重载解析
  9. C# split 几种使用方法
  10. 近来工作和面试一些人的感受(原)