深入解剖指针篇(3)

个人主页(找往期文章) :我要学编程(ಥ_ಥ)-CSDN博客

目录

二级指针

指针数组

指针数组模拟二维数组

字符指针变量 

数组指针

数组指针初始化 

二维数组传参的本质 

函数指针

函数指针的使用

typedef关键字

函数指针数组 


二级指针

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪里?答案是存放在二级指针里头。 指针变量的地址存放在哪里?这句话的主语是地址,问的是地址存放在哪里?我们从前面的学习知识可以知道地址是存放在指针里。只不过这里用了指针变量这个定语来修饰罢了。

上面这个图就是二级指针创建的流程图。

1. *ppa 通过对ppa中的地址进行解引用,这样找到的是 pa , *ppa 其实访问的就是 pa。

2. **ppa 先通过 *ppa 找到 pa ,然后对 pa 进行解引用操作, *pa ,那找到的是 a 。

至于这个**ppa的两颗*的理解:

这个是通过分解成一级指针变量来理解的。

至于ppa+-整数,能访问几个字节,是取决于 int* 的 。这个是指针是4/8个字节。因此ppa+-整数,能访问4/8个字节。

指针数组

指针数组是指针还是数组? 我们类比⼀下,整型数组,是存放整型的数组,字符数组是存放字符的数组。 那指针数组呢?是存放指针的数组。

指针数组的每个元素都是用来存放地址(指针)。如下图:

指针数组的每个元素是地址,又可以指向一块区域。 

指针数组模拟二维数组

 当我们想要打印二维数组的所有元素时,我们是使用下标引用操作符([ ])来实现的。现在学习了指针,那么可以用指针来实现吗?答案是可以的。我们先用一个数组来存放另外几个数组的地址,再通过地址来找到对应的数组,最后再通过打印一维数组的方法来实现。

#include <stdio.h>
void Print(int** p, int sz, int sz1)//arr的类型是int*
{int i = 0;for (i = 0; i < sz; i++){int j = 0;for (j = 0; j < sz1; j++){//printf("%d ", p[i][j]);//下标引用的方法printf("%d ", *(*(p + i) + j));//指针引用的方法}printf("\n");}
}
int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[] = { 2,3,4,5,6 };int arr3[] = { 3,4,5,6,7 };int sz1 = sizeof(arr1) / sizeof(arr1[0]);int* arr[] = { arr1,arr2,arr3 };int sz = sizeof(arr) / sizeof(arr[0]);Print(arr, sz, sz1);return 0;
}

上述的代码模拟出二维数组的效果,实际上并非完全是二维数组,因为每一行并非是连续的。

字符指针变量 

在指针的类型中我们知道有一种指针类型为字符指针 char*。

一般使用方式:

要注意这个ch的内容不可更改,因为这个ch中存放的是一个常量字符串。常量字符串的内容不可更改。但是存放在数组里的字符串可以更改。即数组内容可以更改。

还有一种使用方式如下:

这个 str 中存放的是hello world 的首字符的h,而不是存放hello world 。因为说到底这个str是一个指针变量,存放的是一个地址:这个字符串的地址,然而这个字符串的地址起始就是h的地址,因此就存放的是h的地址。至于在打印这个字符串时,为什么不用解引用?其实是因为str指向的内容就是这个字符串。如果我们还去解引用的话,就会把这个首字符给打印出来。

例如:

《剑指offer》中收录了一道和字符串相关的笔试题,我们一起来观摩⼀下:

#include <stdio.h>
int main()
{char str1[] = "hello bit.";char str2[] = "hello bit.";const char* str3 = "hello bit.";const char* str4 = "hello bit.";if (str1 == str2)printf("str1 and str2 are same\n");elseprintf("str1 and str2 are not same\n");if (str3 == str4)printf("str3 and str4 are same\n");elseprintf("str3 and str4 are not same\n");return 0;
}

答案如下:

我们思考一下是为什么? 首先,str1与str2都是一个地址,在内存中,不可能有两个一模一样的地址。就比如:我们在生活中点外卖,如果在地图上有两个一模一样的地址,那么外卖小哥怎么会知道送去哪一个地方呢?这就产生了错误。因此就打印 str1 and str2 are not same 。接下来就看str3与str4,它们都是一个字符指针,指向的也都是同一个字符串。而我们刚刚知道了这个字符指针存放的是字符串首元素的地址。这个字符串是同一个,那么它们的首元素地址也是一样的。即str3与str4都是指向这个地址,所以str3等于str4,打印 str3 and str4 are same 。(C/C++会把常量字符串存储到单独的一个内存区域,因此str3与str4都是指向这个内存区域)。

数组指针

数组指针是指向数组的指针。我们已经知道了整形指针: int * p;,存放的是整形变量的地址,能够指向整形数据的指针。那数组指针变量应该是:存放的应该是数组的地址,能够指向数组的指针变量。

想要知道答案,首先就得明白 * 和 [ ] 这两个操作符的优先级。

C 运算符优先级 - cppreference.com   这个是运算符优先级和结合性的网址。

由此可知:上面一个是数组,存放整形指针变量的,因此称为指针数组。下面一个是指针(()使p与*先结合——>指针),指向的是一个数组,因此称为数组指针。既然知道是指针了,那么怎么解读这个指针呢?如图所示:

数组指针初始化 

数组指针是指向数组的指针,存放的是数组的地址,那怎么获得数组的地址呢?就是我们之前学习的 &数组名 。这个就是得到的整个数组的地址。

这里可能会有小伙伴有疑惑:这个数组指针的元素个数能不能省略? 答案是不能。因为我们在数组里学过可以省略,但是这个不是数组,而是指针。举例:

二维数组传参的本质 

我们还没有学习指针之前,二维数组传参是这样的:

#include <stdio.h>
void Print(int arr[3][5], int row, int col)//用数组传参的方式接收
{int i = 0;for (i = 0; i < row; i++){int j = 0;for (j = 0; j < col; j++){printf("%d ", arr[i][j]);}printf("\n");}
}
int main()
{int arr[3][5] = { {1,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7} };//打印数组Print(arr, 3, 5);return 0;
}

我们这个是用形参的方式来接收的。如果用指针呢?我们知道数组名是首元素的地址,在二位数组中,我们学过把二位数组看成几个一维数组组成。那么就可以在一维数组的层面把二维数组中的一维数组看成一个一个的元素。那么就可以推出来,在一维数组的层面,二维数组的数组名是首元素的地址,也就是二维数组中第一个一维数组的地址。

#include <stdio.h>
void Print(int(*p)[5], int row, int col)
{int i = 0;for (i = 0; i < row; i++){int j = 0;for (j = 0; j < col; j++){printf("%d ", *(*(p + i) + j));//(p+i)中的p是整个(第一个)一维数组的地址,+1,跳过的是整个一维数组。//因为int(*)[5]是数组指针的类型,这个是+1,就是跳过的。//*p访问的是第一个数组,而*(*p+j)访问的就是第一个数组里的元素。}printf("\n");}
}
int main()
{int arr[3][5] = { {1,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7} };//打印数组Print(arr, 3, 5);//数组名是首元素的地址,也就是整个(第一个)一维数组的地址。//整个(第一个)一维数组的地址要用数组指针来接收。return 0;
}

有的小伙伴可能会疑惑为什么 int(*)[5]是数组指针的类型?这个我在数组知识点这篇文章写过,大家可以去看看。意味着二维数组传参本质上也是传递了地址,传递的是第一行这个一维数组的地址

总结:二维数组传参,形参的部分可以写成数组,也可以写成指针形式。

函数指针

什么是函数指针呢? 根据前面学习整型指针,数组指针的时候,我们可以类比关系,我们不难得出结论: 函数指针应该是用来存放函数地址的,未来通过地址能够调用函数的。我们可以先看看函数的地址是啥样?是不是和数组是一样的?

void test()
{printf("hehe\n");
}
int main()
{printf("%p\n", &test);return 0;
}

看来&函数名是能够把函数的地址地址取出来,那么就看看这个函数名是否能代表函数的地址? 

如此看来,这个函数名也是代表函数的地址。

总结:函数的地址用两种方法可以取出:1.  &函数名   2.   函数名

既然能把函数的地址取出,就肯定要放到函数指针里头。怎么存放呢?

还可以写成这样:

函数指针的使用

通过函数指针调用指针指向的函数。

#include <stdio.h>
int Add(int x, int y)
{return x + y;
}
int main()
{int a = 0;int b = 0;scanf("%d%d", &a, &b);int (*p)(int, int) = Add;int ret = (*p)(3,4);//相当于Add(3, 4);printf("%d\n", ret);
}

然而我们会发现这个代码也是可以的:

这也就意味着这个 * 其实有没有都无所谓。这个函数指针的变量名就相当于这个函数名。

typedef关键字

 typedef 是用来类型重命名的,可以将复杂的类型,简单化。

比如,你觉得 unsigned int 写起来不方便,如果能写成 uint 就方便多了,那么我们可以使用:

typedef unsigned int uint//将无符号整型重新命名为uint,这个就代表无符号整型

如果是指针类型,能否重命名呢?其实也是可以的,比如,将 int* 重命名为 ptr_t ,这样写: 

typedef int* ptr_t

但是对于数组指针和函数指针稍微有点区别。比如我们有数组指针类型 int(*)[5] ,需要重命名为 parr_t ,那可以这样写:

typedef int(*parr_t)[5]; //新的类型名必须在*的右边

函数指针类型的重命名也是一样的,比如,将 void(*)(int) 类型重命名为 pf_t ,就可以这样写: 

typedef void(*pf_t)(int)///新的类型名必须在*的右边

函数指针数组 

函数指针数组是一个数组,用来存放函数指针的。那么函数指针的数组如何定义呢?

p先和 [ ] 结合,说明 p 是数组,数组的内容是什么呢? 是 int (*)(int) 类型的函数指针。 

好啦,这就是C语言深入解剖指针第三篇的全部内容了!下期见!

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

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

相关文章

【IIC,存储器最强总结】

IIC&#xff0c;存储器总结 存储器介绍AT24C02介绍IIC介绍扩展 上拉&#xff0c;下拉&#xff0c;开漏 IIC时序结构 存储器介绍 存储器介绍&#xff1a; RAM易失性存储器/RAM&#xff1a;随机存取存储器&#xff08;英语&#xff1a;Random Access Memory&#xff0c;缩写&…

Linux服务详解

如有错误或有补充&#xff0c;以及任何改进的意见&#xff0c;请在评论区留下您的高见&#xff0c;同时文中给出大部分命令的示例&#xff0c;即是您暂时无法在Linux中查看&#xff0c;您也可以知道各种操作的功能以及输出 如果觉得本文写的不错&#xff0c;不妨点个赞&#x…

基于SSM的个性化旅游攻略定制系统设计与实现(有报告)。Javaee项目。ssm项目。

演示视频&#xff1a; 基于SSM的个性化旅游攻略定制系统设计与实现&#xff08;有报告&#xff09;。Javaee项目。ssm项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xf…

2路DIN2路DO2路AIN远程4GRTU模块钡铼技术S270

钡铼技术的S270远程4G RTU模块是一款高性能的工业级远程终端单元&#xff0c;它支持2路数字输入(DIN)、2路数字输出(DO)以及2路模拟输入(AIN)&#xff0c;并通过4G网络实现数据的远程传输。这种模块的设计旨在满足各种工业自动化和监控需求&#xff0c;特别适用于那些位于偏远地…

电脑/机顶盒/ps3/4/连接老电视(只有AV、S-Video接口)解决方案之HDMI转AV/S-Video转换器HAV

HDMI转AV/S-Video转换器功能 01、将HDMI高清信号经过视频处理转换成AV、S-VIDEO(PAL/NTSC)的视频信号输出 02、将HDMI数字音频&#xff0c;经过DAC数模芯片处理转成模拟立体声输出 03、采用先进的视频处理技术&#xff0c;对图像的亮度&#xff0c;对比度及色彩进行增强处理 04…

Oracle篇—普通表迁移到分区表(第五篇,总共五篇)

☘️博主介绍☘️&#xff1a; ✨又是一天没白过&#xff0c;我是奈斯&#xff0c;DBA一名✨ ✌✌️擅长Oracle、MySQL、SQLserver、Linux&#xff0c;也在积极的扩展IT方向的其他知识面✌✌️ ❣️❣️❣️大佬们都喜欢静静的看文章&#xff0c;并且也会默默的点赞收藏加关注❣…

华为1.24秋招笔试题

华为1.24秋招笔试题 1.题目1 题目详情 - 2024.1.24-华为秋招笔试-第一题-计算积分 - CodeFun2000 1.1题解 import java.util.Scanner;class Main{public static void main(String[] args){Scanner scnew Scanner(System.in);String ssc.next();char[] chs.toCharArray();in…

Vue3.0(一):Vue的引入-options api-模板语法

Vue的引入方式 CDN方式进行引入 将以下 script标签引入即可 <script src"https://unpkg.com/vue3/dist/vue.global.js"></script><!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><met…

AI算力专题:算力存力及汽车电子领先布局,中国封测龙头长奔如电

今天分享的是AI算力系列深度研究报告&#xff1a;《AI算力专题&#xff1a;算力存力及汽车电子领先布局&#xff0c;中国封测龙头长奔如电》。 &#xff08;报告出品方&#xff1a;万联证券&#xff09; 报告共计&#xff1a;27页 全球战略布局完善&#xff0c;多元化、国际化…

TypeScript(十一) 类、对象

1. 类 1.1. 简介 TypeScript是面向对象的JavaScript。   类描述了所创建的对象共同的属性与方法。 1.2. 类的定义 class class_name { // 类作用域 }&#xff08;1&#xff09;定义类的关键字是class&#xff0c;后面紧跟类名&#xff0c;类可以包含以下几个模块&#xff…

数据结构+算法(第03篇):KO!大O——时间复杂度

作者简介&#xff1a;大家好&#xff0c;我是smart哥&#xff0c;前中兴通讯、美团架构师&#xff0c;现某互联网公司CTO 联系qq&#xff1a;184480602&#xff0c;加我进群&#xff0c;大家一起学习&#xff0c;一起进步&#xff0c;一起对抗互联网寒冬 学习必须往深处挖&…

C++进阶--多态

概念 多态是面向对象编程中的一个重要概念&#xff0c;它允许不同类型的对象对同一个消息做出不同的响应。具体的来说&#xff0c;当相同的消息传递给不同的对象时&#xff0c;这些对象能够以不同的方式进行处理&#xff0c;从而产生不同的行为。 对于多态的实现&#xff0c;…