打响指针的第一枪:指针家族

前言

        指针其实是我们学习C语言中最难的知识点,很多人在学习指针的时候会被绕晕,包括博主也是,当初百思不得其解,脑袋都要冒烟了,本来打算在学习指针的时候就写一篇博客,但是当初自己的能力还是没有办法去完成这个壮举,但今时不同往日,如今也算是一名精通C语言的学生了,所以前来编写一篇关于指针的博客。

        本篇博客会让你对指针和数组的了解更深一步,你会发现其实数组和指针并没有什么区别,你也会知道数组指针其实存的就是数组的地址,而数组的地址是比里面元素的地址还要高一级的指针,这里我只会讲解一级数组指针,毕竟指针是可以无限套娃的,讲一个就理解多个了!

        如有表达不清晰或错误,请大家在评论区帮我指正,让我们的学习可以更加完善,而博主也会不断的来更新和修改!

学习目标

  • 首先要搞懂什么是取地址( & ),什么是解引用( * ),以及指针的加法
  • 学习一级指针,二级指针
  • 搞懂 数组指针和指针数组一维、二维数组名、二维数组的行、&数组名

一、理解取地址&、解引用*和指针的加法

        取地址很好理解,就是对一个变量取出它的地址;然后我们要用指针类型来接收这个地址,所以既然指针可以接收地址,那就说明指针就是地址!

        而指针最重要的其实就是解引用和指针的加法,这篇博客会让你理解什么是解引用和指针的加法:深入理解:指针变量的解引用 与 加法运算-CSDN博客

二、快速学习一级指针和二级指针

1. 一级指针

一级指针:其实存的就是非指针变量的地址,可以是各种非指针类型的地址

        而一级指针也是一个变量,变量一定占空间,有空间就要有地址,所以一级指针也是有地址的,千万不能认为一级指针没有地址!!!

    char c = '2';char *pc = &c;            //存char变量的地址short s = 1;short *ps = &s;           //存short变量的地址int i = 3;int *pi = &i;             //存int变量的地址double d = 4.5;double *pd = &d;          //存double变量的地址float f = 5.6f;float *pf = &f;           //存float变量的地址//无符号等基本数据类型struct List l;struct List *plist = &l;  //存结构体stuct 变量的地址union All all; union All *pall = &all;     //存联合体union变量的地址enum、位段等自定义类型

这里面没有涉及对数组的取地址,因为比较特殊,会放在这里讲:🔗

2. 二级指针

二级指针:对一级指针取地址,可以是各种指针类型的地址

所以二级指针就是存放一级指针的地址的指针变量,那同理二级指针也是有地址的,这样就可以实现无限套娃,三级指针、四级指针、n级指针;

    char c;char *pc = &c;            char **ppc = &pc;short s;short *ps = &s;          short **pps = &ps;int i;int *pi = &i;             int **ppi = πdouble d;double *pd = &d;          double **ppd = &pd;float f;float *pf = &f;           float **ppf = &pf;//无符号等基本数据类型struct List l;struct List *plist = &l; struct List **pplist = &plist;union All all;union All *pall = &all;   union All **ppall = &pall;//enum、位段等自定义类型    

三、指针数组

1. 指针数组的介绍

        我们先来学习指针数组的原因就是比数组指针好理解,并且数组名和二维数组的行都是和数组指针有关系的。

        那什么是指针数组呢?

        指针数组,顾名思义:是一个数组,数组元素都是指针类型的,说白了,指针数组就是存放地址的数组。

int arr[5] = {1,2,3,4,5};
int *arr[5] = {arr, arr + 1, arr + 2, arr + 3, arr + 4};

        既然有二级指针、三级指针、四级指针等等,就一定会有一级指针数组、二级指针数组和三级指针数组等等,后面的数组指针也是一个道理,所以我们在这里就仅仅讲解一级指针数组;

int **arr[5];    //二级整型指针数组
char ***ch[5];   //三级字符型指针数组

2. 指针数组的计算

        我们另外一篇文章知道了解引用是根据指针的数据类型(除*之外)来访问字节的;所以直接看下面的例题:

温馨提示:第三个printf语句需要了解大小端字节序才可以解决问题

#include <stdio.h>
int main()
{int arr[5] = {1,40000,3,4,5};int *parr[5] = {arr, arr + 1, arr + 2, arr + 3, arr + 4};printf("%d\n", **parr);printf("%d\n", **(parr + 1));printf("%d\n", *parr[1]);printf("%d\n", *(char*)parr[1]);return 0;
}

第一个printf

首先 **parr ,先看parr 这是一个数组名,是首元素的地址,也就是arr的地址,那parr的数据类型就是int*,*parr解引用是根据 int* 来的,也就是拿出一个指针类型大小的字节(指针类型在32位机器下是4字节,在64位机器下是8字节),取出了arr,那**parr 本质上就是*arr,arr是首元素地址,类型是int*,那*arr解引用就是根据int来的,拿出了一个int类型的大小,4字节,所以**parr = 1;

图解如下:

第二个printf 

同理,这里就是用到了指针+整数,parr的数据类型是int **,那parr + 1,是根据int *来加 的,也就是往后移动一个指针类型的大小,后面的过程都跟第一个相同

图解如下:

第三个printf

就是典型的用下标访问数组元素,但是在这里你就会发现 *(parr + 1) 和 parr[1]是等效的,那我们就可以使用指针的方式和数组下标一起来访问数组元素,因为这是等价的;

图解如下:

第四个printf

这里就涉及到一个强制类型转换,也就会导致我们解引用的时候取出来的字节数是改变的;

具体结果和大小端有关

大端字节序:低地址存放高数字位

小端字节序:低地址存放低数字位

        这里我们能快速地找到parr[1]是arr + 1 这个地址,然后被强制转换为char*类型,这也就表明了解引用的时候,只能取出char类型的字节,1字节。然而这里涉及一个大小端的问题,解引用的时候是从低地址开始解引用,一个字节一个字节取,所以经过char*强转取出来了只有地址最低的一个字节,也就是40;转换为十进制就是64;这是基于小端字节序的结果:

图解如下:

大端字节序的结果为:9c = 156

四、数组名和指针

        终于到了我们的数组名和指针这里了,这里会将数组名和数组指针一起对比着来讲解,大家最好要知道啥是数组指针就行,数组指针就是一个指向整个数组的指针。知道这些我们就开始学习吧!

1. 数组名

我们都知道 数组名表示的是数组首元素的地址,但是有两个特例表示的是整个数组的地址

表示整个数组的地址

  • &数组名
  • sizeof(数组名)

这里想讲解一下 &arr仅仅是一维数组的数组名

int arr[5] = {1,2,3,4,5};

        首先我们知道&arr是整个数组的地址,也就是其数据类型必须是这样: int (*) [5];这也就证实了其实&数组名的本质就是一个数组指针

        那怎样来理解这个类型呢?

        首先我们要让编译器 &arr 知道是整个数组的地址,那就必须让编译器知道有几个数组元素,所以我们会加上[ ],这里大家简单理解一下就行。最后我们只需要知道,&arr表示的是整个数组的地址就行。

接下来看一下下面的题:

#include <stdio.h>
int main()
{int arr[5] = {1,2,3,4,5};printf("%p\n", &arr);printf("%p\n", arr);return 0;
}

运行的结果是什么呢?整个数组的地址是啥样的呢?

        我们惊喜地发现,整个数组的地址居然和数组首元素的地址是一样的,那是真的一样吗?继续看下面的代码:

#include <stdio.h>
int main()
{int arr[5] = {1,2,3,4,5};printf("%p\n", arr);printf("%p\n", arr + 1);printf("%p\n", &arr);    printf("%p\n", &arr + 1);return 0;
}

        我们会发现&arr + 1,跳过了20个字节,也就是5个元素的大小

        所以虽然整个数组的地址和数组首元素的地址是一样的,但是加一之后移动的字节是不同的,本质上是因为数据类型的不同导致的

        arr的数据类型:int * ,&arr的数据类型是 int(*)[5]

2. 二维数组名

二维数组名,同样也适用于对数组名的规则;

先说结论:二维数组的数组名 == 二维数组第一行的地址

下面代表的运行结果是什么呢?

#include <stdio.h>
int main()
{int arr[2][2] = {{1,2},{3,4}};printf("arr: %p\n", arr);printf("arr + 1: %p\n", arr + 1);return 0;
}

        结果是移动了8个字节,也就是两个int类型的大小啊,两个int类型的大小不就是第一行吗?所以通过这个现象可以知道,二维数组的首元素是整个第一行,所以二维数组的数组名就是整个第一行的地址啊!

3. 二维数组的行

二维数组的行:是表示该行这个一维数组的数组名,是该行首元素的地址

        讲解二维数组的行之前

        大家先想一下一维数组的每个元素是什么?

        通过一维数组能不能推出二维数组的每个元素是什么呢?

int arr[2][2] = {{1,2},{3,4}};

        不难想出,二维数组的每个元素其实就是每一行的一维数组,因为上面也隐含了二维数组的数组名是第一行的地址,而数组名又是首元素的地址,那就侧面印证了二维数组其实就是一维数组的数组。但是这跟我们的行有什么关系呢?接下来就是要学习的知识了

        大家先理清一下思路,二维数组的行是什么?二维数组的行就是第一个方括号[ ],而我们要访问一个一维数组元素的时候,是这样访问的:

int a[5] = {1,2,3,4,5};
a[1] = 8;

访问二维数组的第一行的元素,是这样访问的:

int arr[2][2] = {{1,2},{3,4}};
arr[0][1] = 8;

它们之间的共同之处: 都要用数组名+下标引用

一维数组:arr + [1]

二维数组:arr[0] + [1]

所以我们会发现二维数组的行,其实就相当于一维数组的数组名!既然二维数组的行相当于一维数组的数组名了,那就是首元素的地址,arr[0] == &arr[0][0]!

我们学完这些,根本上来说二维数组就可以相当于一级数组指针的数组了!

数组和指针拓展知识

  • a[ i ] = *( a + i )
  • b[ x ][ y ] = *( b[ x ] + y ) = *( *( b + x ) + y )

五、数组指针

终于来到数组指针了!!!!

数组指针,顾名思义:是一种指向数组的指针

我们只讨论一级数组指针,多级数组指针大家有兴趣可以私信我

        我们来思考这样一个问题,既然一个指针是可以指向整个数组的,并且指针是存放地址的变量,那数组指针是如何做到指向整个数组的呢?

        其实不难理解,我只要存放整个数组的地址就可以了呀,那如何存放数组的地址呢?别忘了&数组名代表的就是整个数组的地址哦!而&数组名中的数组名是首元素地址,但是对首元素地址取地址,那不就相当于一个二级指针了吗?既然这样,数组指针就相当于一个二级指针,那二维数组名,实际上也相当于一个二级指针,这也是为什么数组指针和二维数组名拿元素要解引用两次的原因

        但是解引用要涉及到数据类型,那数组的数据类型又是什么呢?

int arr[5] = {1,2,3,4,5};
int (*parr) [5] = &arr;

        提到这里就不得不拓展一个知识点,int arr[5]是一个数组,那这个数组的类型是什么呢?大多数人都没有去研究过吧,我们不妨可以通过以往的经验来看

        比如 int a 的 a是一个变量名,a 的类型是 int ,double d 的 d是一个变量名,类型是double;那int arr[ 5 ]的变量名是什么呢?没有变量名,一定有数组名!所以数组名是arr,那数据类型是int [5] ;这表示这个变量arr是一个数组类型,是一个有5个int类型的数组。

        因为解引用和加法是涉及到类型的问题,所以我们必须要明白数组指针的数据类型是什么,虽然我上面说了数组指针是相当于二级指针的,但是仅仅是为了让我们来理解 解引用2次的原因。

        那到底数组指针的数据类型到底是什么呢?

        首先依旧是拿出数组名parr,剩下的就是数据类型:int (*) [5],这个的意思就是为一个数组的指针类型,但是这里还有数组元素的个数,只有知道元素个数,解引用的时候才知道拿出来多少字节,加的时候才知道移动多少字节。

        *表示这是一个指针,int 表示 元素类型,而[5]表示有多少个元素;

对于加法:数组指针移动的是整个数组的大小;

对于解引用:作者目前没有搞懂深层,但是有一种方法简单易懂:

        因为parr 是 &arr,那 *parr 就是 *&arr, * 和 & 相互抵消了,就是arr,这样我们也就是可以理解为啥是指针降级了。

        所以*parr == arr,那对*parr的解引用或者是加法,就是对arr来的。

六、总结

        其实我们对指针和数组这里的考点基本都是在解引用和指针➕整数这里出题,因为对于学C的大家,这里算是难题了,它往往可以伴随着强制类型转换,隐式类型转换和大小端字节序等多方面出题,但是万变不离其宗,你只要弄清是啥数据类型就OK,仔细画图就一目了然了。

        最后给大家推荐一下我的C语言刷选择题的专栏,这里是我在牛客网上精选出来的题,里面有我的个人解析,如有错误,请大家指正,有不懂的不会的可以私信哦!

https://blog.csdn.net/2302_76941579/category_12492707.html?spm=1001.2014.3001.5482

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/285016.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【具身智能评估7】ProcTHOR: Large-Scale Embodied AI Using Procedural Generation

论文标题&#xff1a;ProcTHOR: Large-Scale Embodied AI Using Procedural Generation 论文作者&#xff1a;Matt Deitke, Eli VanderBilt, Alvaro Herrasti, Luca Weihs, Jordi Salvador, Kiana Ehsani, Winson Han, Eric Kolve, Ali Farhadi, Aniruddha Kembhavi, Roozbeh M…

【TI毫米波雷达】上电时序、串口回环BUG及SOP模式不正常工作的解决方案(LP87524电源PMIC芯片的BUCK供电时序配置)

【TI毫米波雷达】雷达上电时序及SOP模式不正常工作的解决方案&#xff08;LP87524电源PMIC芯片的BUCK供电时序配置&#xff09; 文章目录 上电时序上电以后的雷达串口回环问题延迟上电时序LP87524电源PMIC芯片的BUCK供电时序LP87524电源PMIC芯片的BUCK默认供电输出附录&#x…

早期的OCR是怎么识别图片上的文字的?

现在的OCR技术融合了人工智能技术&#xff0c;通过深度学习&#xff0c;无论是识别的准确率还是效果都非常不错&#xff0c;那您知道在早期的OCR是通过什么技术来实现的吗&#xff1f;如果您不知道&#xff0c;那么&#xff0c;就让我来告诉您&#xff1a;它主要是基于字符的几…

LibreNMS:从docker出发

引言 LibreNMS 是一个免费开源的网络监控和自动化工具&#xff0c;用于监视网络设备、服务器和应用程序的性能和状态。它提供了一个集中的管理平台&#xff0c;帮助管理员实时监控和管理整个网络基础设施。 以下是 LibreNMS 的一些主要特点和功能&#xff1a; 自动发现&#…

【.NET】控制台应用程序的各种交互玩法

关于控制台交互&#xff0c;大伙伴们也许见得最多的是进度条&#xff0c;就是输出一行但末尾不加 \n&#xff0c;而是用 \r 回到行首&#xff0c;然后输出新的内容&#xff0c;这样就做出进度条了。不过这种方法永远只能修改最后一行文本。 于是&#xff0c;有人想出了第二种方…

小程序地图检索

<template><view style"background-color: #f5f5f5"><!-- 头部开始 --><viewstyle"position: fixed;left: -5rpx;right: -5rpx;z-index: 99;top: -5rpx;width: 101vw;"><view style"position: relative"><view…

三大主流前端框架介绍

在前端项目中&#xff0c;可以借助某些框架&#xff08;如React、Vue、Angular等&#xff09;来实现组件化开发&#xff0c;使代码更容易复用。此时&#xff0c;一个网页不再是由一个个独立的HTML、CSS和JavaScript文件组成&#xff0c;而是按照组件的思想将网页划分成一个个组…

Go集成elasticsearch8极简demo,光速入门

Go集成elasticsearch8极简demo,光速入门 配置go环境创件go mod工程代码实现配置go环境 编辑器添加goproxy GO111MODULE=on;GOPROXY=https://mirrors.wps.cn/go/,https://goproxy.cn,direct;GOSUMDB=off创件go mod工程 mkdir demo cd demo go mod init demo代码实现 在demo…

人工智能与底层架构:构建智能引擎的技术支柱

导言 人工智能与底层架构的交融塑造了智能系统的基石&#xff0c;是推动智能时代发展的关键动力&#xff0c;本文将深入研究人工智能在底层架构中的关键作用&#xff0c;以及它对智能引擎的技术支持&#xff0c;探讨人工智能在计算机底层架构中的作用&#xff0c;以及这一融合如…

MongoDB的数据库引用

本文主要介绍MongoDB的数据库引用。 目录 MongoDB的数据库引用 MongoDB的数据库引用 MongoDB是一种面向文档的NoSQL数据库&#xff0c;它使用BSON&#xff08;Binary JSON&#xff09;格式存储和查询数据。在MongoDB中&#xff0c;数据库引用是一种特殊的数据类型&#xff0c;…

cefsharp120.1.8(cef120.1.8,Chromium120.0.6099.109)版本升级测试,其他版本H264版本

此版本最新版cef120.1.8,Chromium120.0.6099.109 此更新包括一个高优先级安全更新 This update includes a high priority security update. 说明&#xff1a;本版本暂时不支持264&#xff0c;其他H264版本参考119,116&#xff0c;114&#xff0c;110&#xff0c;109等版本 c…

【贪心算法】之跳跃游戏

问题&#xff1a; 给定一个非负整数数组 [nums] &#xff0c;开始时位于数组的初始位置 &#xff0c;数组中每个下标对应的元素代表你在该位置可以跳跃的最大长度。 判断你是否能够到达数组的最后位置。**思路&#xff1a;**贪心算法 看不懂的可以去下面这个链接看 具体思路 …