【C语言】指针的进阶(一)

news2025/1/10 10:43:04

目录

前言

1. 字符指针

2. 指针数组

3. 数组指针

3.1 数组指针的定义

 3.2 &数组名VS数组名

 3.3 数组指针的使用

4. 数组参数、指针参数

4.1 一维数组传参

 4.2 二维数组传参

 4.3 一级指针传参

 4.4 二级指针传参

5. 函数指针


 

前言

指针在C语言中可谓是有着举足轻重的存在,初学C语言的我们在《指针》章节已经接触过了一些指针的知识,知道了指针的概念:

  1. 指针就是个变量,用来存放地址,地址唯一标识一块内存空间。
  2. 指针的大小是固定的4/8个字节(32位平台/64位平台)。
  3. 指针是有类型,指针的类型决定了指针的+-整数的步长,指针解引用操作的时候的权限。
  4. 指针的运算。

 指针的基础知识已经了解完毕,那么在这一篇博客里,我们将深入探讨指针的一些高级使用。

1. 字符指针

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

 一般使用:

int main()
{
	char ch = 'w';
	char* pc = &ch;
	return 0;
}

 还有一种使用方式:

int main()
{
	const char* pstr = "abcdef";
	printf("%s\n", pstr);
	return 0;
}

上面这种用法很多人会有一个误区,以为是把字符串 abcdef 放到字符指针 pstr 里了,但是本质是把字符串 abcdef 中的首字符的地址放到了pstr中。

下面可以试着证明一下

  • 字符串"abcdef"的地址就是a所在地址,那么"abcdef"[3]相当于 ““a地址[3]”” ,侧面印证了确实是把首字符地址存入了指针pstr中。
  • 数组名就是首元素地址,既然说字符指针存放的是首字符的地址,那么试着用数组下标的方式访问字符指针指向的内容,结果发现一样可以打印出来。
  • 因此完全可以把常量字符串想象成一个数组,然后用字符指针接收,操作起来与数组一致。

注意:最好使用 const 修饰存放字符串的字符指针,因为字符串是常量,不允许被修改,如果修改了程序会崩溃。

一道经典【面试题】,出自《剑指offer》:

以下的最终输出是什么呢? 

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

	if (str3 == str4)
		printf("str3 and str4 are same\n");
	else
		printf("str3 and str4 are not same\n");

	return 0;
}

【答案】

str1 and str2 are not same

str3 and str4 are same

【解释】

  • str1 和 str2 :相同的常量字符串初始化不同的数组的时候就会开辟出不同的内存块,str1和str2其实是各自创建了一个空间存放nash.,因此它们的地址是不一致的。
  • str3 和 str4 :当用指针指向字符串时,因为nash是常量字符串,是不会被修改的,那么既然不能被修改,编译器就没必要保存多份,只需要一份,然后让指针都指向同一块内存即可,因此地址值是一致的。

【扩展】

int main()
{
	char str1[] = "hello bit.";
	char str2[] = "hello bit.";
	const char* str3 = "hello bit.";
	const char* str4 = "hello bit.";

	if (&str3 == &str4)
		printf("Yes");
	else
		printf("No");
	return 0;

【答案】

 结果为No,因为str3和str4本身的地址值是不一样的,只是str3和str4指向的内容是一样的。

2. 指针数组

这里我们复习一下,下面指针数组是什么意思?

int* arr1[10]; //整形指针的数组
char *arr2[4]; //一级字符指针的数组
char **arr3[5];//二级字符指针的数组

类比一下:

字符数组 —— 存放字符的数组

整型数组 —— 存放整形的数组

那么:

指针数组 —— 存放指针的数组,即存放在数组中的元素都是指针类型

 那么很多人有个疑问,指针数组到底有什么用呢?

很多人直观的感受就是定义abcd,然后将它们的地址都存入整型指针数组中。如下:

错误的使用方式,【没有意义】
int main()
{
	int a = 0;
	int b = 1;
	int c = 2;
	
	int* arr[3] = { &a,&b,&c };
	return 0;
}

但是很少有人会这样使用的,没有这种使用场景,也没有人会这样使用,这样写的没有任何意义。

正确的使用场景之一:

【可以使用指针数组模拟一个二维数组】
int main()
{
	int arr1[] = { 1,2,3,4,5 };
	int arr2[] = { 2,3,4,5,6 };
	int arr3[] = { 3,4,5,6,7 };
	
	int* arr[3] = { arr1,arr2,arr3 };
    //遍历三个数组
	int i = 0;
	for ( i = 0; i < 3; i++)
	{
		int j = 0;
		for ( j = 0; j < 5; j++)
		{
			printf("%d ", arr[i][j]);
		}
		printf("\n");
	}
	return 0;
}

 使用指针数组维护多个数组,模拟出一个二维数组,因此操作也与而二维数组类似。 

 还有使用指针数组维护多个字符串:

 当然还有很多其他的应用场景,但是由于篇幅有限,这里就不再一一列举。

3. 数组指针

3.1 数组指针的定义

数组指针是指针?还是数组?

类比一下:

整型指针 —— 指向整形的指针

字符指针 —— 指向字符的指针

那么:

数组指针 —— 指向数组的指针

下面代码哪个是数组指针?

int *p1[10];
int (*p2)[10];
//p1, p2分别是什么?

 【解释】

p2是数组指针。

p2先和*结合,说明p2是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个指针,指向一个数组,叫数组指针。
这里要注意:[ ]的优先级要高于*号的,所以必须加上()来保证p2先和*结合。

 3.2 &数组名VS数组名

往期博客(链接:点击前往)中有提到过:

数组名就是地址,通常来说:数组名是数组首元素的地址。

但是,存在两种特殊情况:

1、sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节。

2、&数组名,这里的数组名表示整个数组,取出的是整个数组的地址。

除此之外,所有遇到的数组名都是数组首元素的地址。

 由上图可以证明,第一组和第二组+1时都只是跳过四个字节,即arr表示的是首元素地址。

而第三组可以发现,&arr和&arr+1之间跳过了40个字节,即一整个数组的大小,因此证明&arr表示整个数组。

指针类型决定了指针+1到底+几个字节。

那么我们大胆猜测一下,arr的类型是int*,&arr[0]的类型也是int*,

那么&arr的类型就是int (*)[10],即数组指针存放数组。

 既然看完了什么是数组指针,那么做一道题来检测一下自己是否理解明白。

【练习】

下面代码中pc的类型是什么?

int main()
{
	char* arr[5];
	pc = &arr;  //给pc定义类型

	return 0;
}

 【答案】

char* (*pc)[5] = &arr; 

【解释】

首先pc得是个指针,即(*pc)

指向一个元素为5的数组,即(*pc)[5]

数组每个元素的类型是char*,即char* (*pc)[5]

那么数组指针有哪些使用场景呢?

 3.3 数组指针的使用

错误使用场景:

 这种场景一点也不方便,在使用数组时反而更麻烦了,有一种 “脱裤子放屁” 的感觉。这种使用方式基本上不会出现。

如果非要用指针接收数组,也应该使用指针接收首元素地址,而不是接收整个数组的地址。这才是正确的访问姿势。

其实,在一维数组传参过程中,形参既可以写成数组的形式,又可以写成指针的形式,因为本质上这两种方式都是传递的是数组首元素地址,是互等的。

既然本质上都是地址,那么为什么还要写成数组形式呢?

那是因为能够写成数组形式完全是为了照顾初学者,因为实参是一个数组,形参也定义一个数组来接收这个数组这种方式更能够让初学者理解。因此即使这么写,本质也还是指针。

同理,二维数组传参也有两种方式:

  1. 形参也是使用二维数组的形式。
  2. 形参使用数组指针的形式。

学了指针数组和数组指针我们来一起回顾并看看下面代码的意思:

int arr[5];
int *parr1[10];
int (*parr2)[10];
int (*parr3[10])[5];

【答案】

  1. arr 是一个能够存放5个整型数据的数组。
  2. parr1 是一个数组,能够存放整型指针的数组,数组大小为10。
  3. parr2 是一个指针,指向元素为10的数组,每个元素的类型是int*
  4. parr3 是一个数组,数组有10个元素,指向一个指针,指针指向的内容是元素为5的数组,每个元素的类型是int,即是一个存放数组指针的数组。

当然parr3看不懂也不要着急,这种形式是很少用到的,只是作为拓展知识点了解即可。

4. 数组参数、指针参数

4.1 一维数组传参

 4.2 二维数组传参

二维数组传参,函数形参的设计只能省略第一个[]的数字。因为对一个二维数组,可以不知道有多少行,但是必须知道一行多少元素。这样才方便运算。

 4.3 一级指针传参

 一级指针传参,形参就写成一级指针就行。

#include <stdio.h>
void print(int* p, int sz)
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		printf("%d\n", *(p + i));
	}
}
int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9 };
	int* p = arr;
	int sz = sizeof(arr) / sizeof(arr[0]);
	//一级指针p,传给函数
	print(p, sz);
	return 0;
}

 【思考】

当一个函数的参数部分为一级指针的时候,函数能接收什么参数?

【答案】

 4.4 二级指针传参

#include <stdio.h>
void test(int** ptr)
{
	printf("num = %d\n", **ptr);
}
int main()
{
	int n = 10;
	int* p = &n;
	int** pp = &p;
	test(pp);
	test(&p);
	return 0;
}

【思考】

 当函数的参数为二级指针的时候,可以接收什么参数?

【答案】

5. 函数指针

函数指针 —— 指向函数的指针 —— 存放的是函数的地址

 &Add和Add完全等价,下面的例子可以证明。

可以定义 int (*pf2)(int,int) = &Add; 将&Add赋值给pf2, pf2便是函数指针

  • Add引用函数返回ret1
  • 此时对pf2解引用在调用函数返回ret2
  • 对pf2直接调用函数返回ret3
  • 在pf2前面加很多解引用返回ret4

最后打印结果居然发现,四个结果都是一样的。

所以得出结论:函数指针在调用所指向函数时,可以不写*直接和函数名一样调用函数,而*号在这里其实就只是一个摆设,同样是为了照顾初学者的使用习惯,所以才会导致当加了很多*号去解引用时得出来的结果依然是正确的结果。

 下面来看两端有趣的代码:

1、

(*(void (*)())0)();
  •  void (*)()              ———— 函数指针类型
  • void (*)() )0         ———— 对0进行强制类型转换,转换成函数指针。即将地址0当做存放函数指针的地址。
  • (*( void (*)() )0 )()  ———— 通过函数指针调用函数,函数的参数为空。

即上面的代码实在调用0地址处的函数,这个函数没有参数,返回值是void。

 该代码出自《C陷阱和缺陷》

2、

void (*signal(int , void(*)(int)))(int);
  • signal(int, void(*)(int))  ———— signal是一个函数,它有两个参数,整型int和函数指针类型void(*)(int)。
  • void (*signal(int , void(*)(int)) )(int); ———— signal函数的返回类型也是函数指针类型void (*)(int)

但是这个代码看着太复杂了,有没有办法简化呢?

有没有办法将代码写成符合我们习惯的形式这样:返回类型在前,函数名在中间,函数参数在后的:void (*)(int) signal(int , void(*)(int)),这样直接写是肯定不支持的,但是可以通过typedef优化一下:

typedef void(*pfun_t)(int);  //对void(*)(int)重新起名为pfun_t
pfun_t signal(int, pfun_t);

对void(*)(int)重新起名为pfun_t,这样写出来的代码就清楚多了。

 


 如果觉得作者写的不错,求给作者一个大大的点赞支持一下,你们的支持是我更新的最大动力!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL夺命20连问

博主介绍&#xff1a;✌全网粉丝3W&#xff0c;全栈开发工程师&#xff0c;从事多年软件开发&#xff0c;在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战&#xff0c;博主也曾写过优秀论文&#xff0c;查重率极低&#xff0c;在这方面有丰富的经验…

Seata 源码篇之核心思想 - 01

Seata 源码篇之核心思想 - 01 引言基础架构数据源代理分支事务提交和回滚隔离级别解决脏写读未提交读已提交 小结 笔者个人项目中使用到了seata来做分布式事务管理&#xff0c;面试过程中也经常被问到seata的原理&#xff0c;seata源码本身也不是很复杂&#xff0c;所以准备出一…

FWT小结

核心思想&#xff1a;把 a , b a,b a,b 化成 f w t ( a ) , f w t ( b ) fwt(a),fwt(b) fwt(a),fwt(b)&#xff0c;相乘后再化为 a a a 化的过程用的是分治 所以和FFT其实一模一样 OR / AND 卷积 不需要什么技巧&#xff0c;暴力分治转移即可 每次分治下去&#xff0c;…

瑞萨MCU入门教程(非常详细的瑞萨单片机入门教程)

瑞萨MCU零基础入门系列教程 前言 得益于瑞萨强大的MCU、强大的软件开发工具(e studio)&#xff0c;也得益于瑞萨和RA生态工作室提供的支持&#xff0c;我们团队编写了《ARM嵌入式系统中面向对象的模块编程方法》&#xff0c;全书37章&#xff0c;将近500页: 讲解面向对象编程…

硬件笔记:组装“固态 U 盘”的八年,从 100 块到 1000 块

这篇文章&#xff0c;聊聊自从 2015 年开始&#xff0c;到目前为止&#xff0c;我使用固态硬盘组装的高速 U 盘&#xff0c;以及它们的使用体验&#xff0c;以及一些明显的坑。 写在前面 2015 年的 8 月&#xff0c;我剁手下单了一块 32G 大小&#xff0c;NGFF接口的三星 22x…

关于 C/C++ 中在指针前加 const 关键字的作用说明

1. 作用说明&#xff1a; 在指针前加 const 的用途为&#xff1a;不可改变指针指向的内存的值&#xff0c;即将该指向指向的内存中的变量置为只读&#xff08;read-only) 变量。 但是&#xff0c;可以给 const 的指针赋值&#xff0c;即将具有 const 属性的指针指向别的内存地…

Linux 内核镜像分析

文章目录 前言一、概述二、bzImage2.1 镜像分析 三、zImage3.1 镜像分析参考链接 前言 介绍了vmlinux的来源&#xff0c;以及分析方法&#xff1b; 一、概述 在linux系统中&#xff0c;vmlinux&#xff08;vmlinuz&#xff09;是一个包含linux kernel的静态链接的可执行文件…

JavaScript中的原型继承和类继承之间的区别

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ 原型继承&#xff08;Prototype Inheritance&#xff09;⭐ 类继承&#xff08;Class Inheritance&#xff09;⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 记得点击上方或者右侧链接订阅本专栏哦 几何带你启…

MySQL开启安全审计日志,开启查询日志

MySQL 查询开启日志 在 MySQL 数据库中&#xff0c;开启查询日志是一个非常有用的技术&#xff0c;它能帮助你追踪每一个执行的查询语句&#xff0c;以便更好地优化 SQL 语句和性能。本文将介绍如何在 MySQL 数据库中开启查询日志。 开启查询日志 MySQL 中的查询日志是一种记…

Mysql开启binlog

本案例基于mysql5.7.16实验 1、在linux中进入mysql查询binlog是否打开&#xff0c;执行命令如下&#xff1a; mysql -u root -p 2、查询binlog是否开启命令如下&#xff0c;如果log_bin为OFF则证明mysql的binlog没有打开 show variables like %log_bin%; 3、退出mysql终端&…

OPC DA如何实现跨平台

目录 简介 EntireX DCOM Utgard OPC XML DA OPC UA 协议转换代理 简介 本文介绍OPC DA跨平台通讯的几种方案。 OPC官方说明文档 OPC&#xff08;OLE for Process Control&#xff09;是为过程控制专门设计的OLE 技术&#xff0c;基于COM/DCOM的数据访问的标准。常说的O…

vmware去虚拟化

路径&#xff1a;C:\Program Files (x86)\VMware\VMware Workstation\x64\vmware-vmx.exe &#xff0c;复制一份备份 用16进制工具打开修改这个文件&#xff0c;如winhex 1、搜索 25 73 2E 65 6E 61 62 6C &#xff0c;找到上面有两个"VMware"开头的 2、硬盘SCSI格…

JAVASE---String类

String类的重要性 在C语言中已经涉及到字符串了&#xff0c;但是在C语言中要表示字符串只能使用字符数组或者字符指针&#xff0c;可以使用标准库提供的字符串系列函数完成大部分操作&#xff0c;但是这种将数据和操作数据方法分离开的方式不符合面相对象的思想&#xff0c;而…

Java 类和对象

在面向对象语言中万物皆对象&#xff0c;一切都围绕对象来进行&#xff0c;找对象、建对象&#xff0c;用对象等。 类&#xff1a;把具有相同特征和行为的一组对象抽象为类&#xff0c;类是抽象概念&#xff0c;如人类、车类等&#xff0c;无法具体到每个实体。 对象&#xff…

71、Spring Data JPA 的 样本查询--参数作为样本去查询数据库的数据,也可以定义查询的匹配规则

★ 样本查询 给Spring Data传入一个样本数据&#xff0c;Spring Data就能从数据库中查询出和样本相同的数据。被查询的数据并不需要和样本是完全相同的&#xff0c;可能只需要和样本有几个属性是相同的。总结&#xff1a; 样本查询–就是把参数作为样本去查询数据库的数据&…

一、 计算机网络概论

一、计算机网络概论 1、计算机网络概述 1.1、概念 计算机网络是一个将分散的、具有独立功能的计算机系统&#xff0c;通过通信设备与线路连接起来&#xff0c;由功能完善的软件实现资源共享和信息传递的系统 是一些互连的、自治的计算机系统的集合 以能够相互共享资源的方…

【Git】Git 变基(rebase)以及rebase和merge之间的区别

Git 变基 1.变基 — rebase 在 Git 中整合来自不同分支的修改主要有两种方法&#xff1a;merge 以及 rebase。 在前面的文章中已经介绍了merge&#xff0c;这里我们来学习另一个指令rebase。 变基的基本操作 回顾之前在 分支的合并 中的一个例子&#xff0c;在该例子中&am…

go Gorm连接数据库,并实现增删改查操作

Gorm 1. 准备工作 首先进入终端下载我们需要的包&#xff08;确保go和mysql安装完成&#xff0c;并设置了环境变量&#xff09; go get -u gorm.io/driver/mysql go get -u gorm.io/gorm有两份官方文档有对 GORM 更详细的讲解。 创建 | GORM - The fantastic ORM library f…

EndNote21 | 账户同步问题

问题&#xff1a;无法同步&#xff0c;提示如下图所示。 原因&#xff1a;网络问题。 解决方法&#xff1a;国内网络无法实现同步&#xff0c;解决上网问题即可。

【数据结构】结构实现:顺序存储模式实现堆的相关操作

&#x1f6a9;纸上得来终觉浅&#xff0c; 绝知此事要躬行。 &#x1f31f;主页&#xff1a;June-Frost &#x1f680;专栏&#xff1a;数据结构 &#x1f525;该文章着重讲解了使用顺序结构实现堆的插入和删除等操作。 目录&#xff1a; &#x1f30d;二叉树的顺序结构&#x…