C语言指针进阶:各类型指针变量详解

目录

  • 1. 字符指针变量
  • 2. 数组指针变量
    • 2.1 什么是数组指针变量
    • 2.2 数组指针变量的初始化
  • 3. 二维数组传参的本质
  • 4. 函数指针变量
    • 4.1 函数指针变量的创建
    • 4.2 函数指针变量的使用
    • 4.3 代码分析
      • 4.3.1 typedef 关键字
  • 5. 函数指针数组
  • 6. 转移表


正文开始。

1. 字符指针变量

我们可以通过指针来指向一个字符变量
例如:

int main()
{
	char ch = 'a';
	char* pch = &ch;
	return 0;
}

还可以这样:

int main()
{
	const char* pstr = "hello world!";
	return 0;
}

上述第三行代码,很容易让人理解为是把字符串hello world!放到了字符指针pstr里了。但其实这里本质是把字符串hello world!首字符的地址放到了pstr里。

值得注意的是:当几个指针指向同一个字符串的时候,他们实际会指向同一块内存。这也不难理解,某一字符串是独一无二的,没必要为同一个东西再开辟空间。

例如:

#include <stdio.h>
int main()
{
	char str1[] = "hellw world!";
	char str2[] = "hellw world!";
	const char *str3 = "hellw world!";
	const char *str4 = "hellw world!";
	
	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;
}

运行结果:
在这里插入图片描述
从这里就可以看出,几个指针指向同一个字符串的时候,他们会指向同一块内存;但用相同的字符串去初始化不同的数组时,就会开辟处不同的内存块。

2. 数组指针变量

2.1 什么是数组指针变量

在上一篇文章中,我们学习了指针数组,它是存放指针的数组。
类比指针数组,不难理解,数组指针变量就是指向数组的指针变量

下面我们来看两种变量:

//指针数组 - 存放指针的数组
int* p1[10];

//数组指针变量 - 指向数组的指针变量
int (*p2)[10];

理解:

  • *的优先级低于[]
  • int* p1[10]p1先与[10]结合,代表数组类型,int *指明数组中元素的类型为整型指针
  • int (*p2)[10]p2先与 * 结合,代表指针类型,int [10]代表所指向对象的类型为存放了十个整型元素的数组

2.2 数组指针变量的初始化

数组指针变量是用来存放数组地址的,我们可以通过取地址操作符&来获取数组地址。

int main()
{
	int arr[10] = { 0 };
	//数组指针变量的初始化
	int (*p)[10] = &arr;
	return 0;
}

3. 二维数组传参的本质

在我们将一个二维数组传递给一个函数时,我们是这样写的:

#include <stdio.h>

void Print(int arr[2][3], int row, int col)
{
	int i = 0;
	int j = 0;
	for(i = 0; i< row; i++)
	{
		for(j = 0; j < col; j++)
		{
			printf("%d ",arr[i][j]);
		}
		printf("\n");
	}
}

int main()
{
	int arr[2][3] = {{1,2,3}, {2,3,4}};
	Print(arr,2,3);
	return 0;
}

我们再重新理解下二维数组,二维数组定义时,通常像这样int arr[2][3],这里我们可以看作是这样的int (arr[2])[3],即一维数组里面存放的元素是一维数组,这样就把一个二维数组拆分成了两个一维数组来看。例如第一行的一维数组的类型就是int [3],所以第一行的地址的类型就是数组指针类型int(*)[5]。通过这一点,我们不难理解,二维数组传参本质上也是传递了参数,传递的是第一行这个一维数组的地址

那么,我们二维数组传参,也可以写成这样:

#include <stdio.h>

void Print(int (*p)[3], int row, int col)
{
	int i = 0;
	int j = 0;
	for(i = 0; i < row; i++)
	{
		for(j = 0; j < col; j++)
		{
			printf("%d ", *(*(p + i) + j));
			//p + i == &arr[i]
			//*(p + i) == arr[i] 相当于第i行一维数组的数组名
			//*(p + i) + j == &arr[i][j]
			//*(*(p + i) + j) == arr[i][j]
		}
		printf("\n");
	}
}
int main()
{
	int arr[2][3] = {{1,2,3}, {2,3,4}};
	Print(arr,2,3);
	return 0;
}

上述代码运行结果:
在这里插入图片描述

4. 函数指针变量

函数指针变量,顾名思义,它就是存放函数地址的指针变量

在学习函数指针变量前,我们要了解到,函数名就是函数的地址。

例如:

#include <stdio.h>
//函数地址 -- &Add
//函数地址 -- Add
int Add(int x, int y)
{
	return x + y;
}
int main()
{
	printf("&Add = %p\n", &Add);
	printf("Add  = %p\n", Add);
	return 0;
}

上述代码运行结果:
在这里插入图片描述

4.1 函数指针变量的创建

当我们想把函数的地址存放起来,这是就需要用到函数指针变量了,函数指针变量的写法与数组指针非常类似。例如:

#include <stdio.h>

int Add(int x, int y)
{
	return x + y;
}
int main()
{
	int (*p1)(int, int) = Add;
//或int (*p2)(int x, int y) = Add;
//Add也可写为&Add,它们都代表函数的地址

//int    (*p1)     (int, int)
// |       |       ——————————
// |       |            |
// |       |            |
// |       |       p1指向函数的参数类型和个数的声明
// |       函数指针变量名
// p1指向函数的返回类型
//
// p1的类型 -- int (*) (int x, int y)
	return 0;
}

4.2 函数指针变量的使用

我们可以通过函数指针调用指针指向的函数

#include <stdio.h>

int Add(int x, int y)
{
	return x + y;
}
int main()
{
	//函数指针变量定义
	int (*p1)(int, int) = Add;
	//函数指针变量使用
	printf("%d\n",(*p1)(3,4));
	printf("%d\n", p1(5,1));
	return 0;
}

上述代码运行结果:
在这里插入图片描述

4.3 代码分析

我们来看下面两个代码

代码1:

(*(void (*)())0))();
//void (*)() -- 函数指针类型,所指向对象无形参,无返回值

//(void (*)())0 -- 将0强制转换类型为void (*)()类型,即将0地址处存放函数的地址

//*(void (*)())0) -- 解引用函数指针

//(*(void (*)())0))() -- 调用函数指针变量指向的函数

代码2:

void (*signal(int, void(*)(int)))(int);
//void(*)(int) -- 函数指针变量,指向的函数无返回值,参数类型为int

//signal(int, void(*)(int)) -- 函数名为signal的函数,第一个参数类型为int,第二个参数类型为void(*)(int)

//void (*signal(int, void(*)(int)))(int) -- 函数signal(int, void(*)(int))的声明,它的返回值类型为void (*)(int)

4.3.1 typedef 关键字

上面我们写的代码二的作用就是声明一个函数,可以看出来,这个函数声明非常的复杂,我们可以通过typedef关键字来重命名类型,简化类型

例如:

typedef int i;
//将 int 重命名为 i

typedef unsigned int uint;
//将 unsigned int 重命名为 uint

typedef int(*parr)[5];
//将 int (*)[5] 重命名为 parr

typedef void(*pfun)(int);
//将 void(*)(int) 重命名为 pfun

若要简化代码2,我们可以这样写:

typedef void(*pfun)(int);
//将 void(*)(int) 重命名为pfun

pfun signal(int, pfun);

5. 函数指针数组

函数指针数组,就是存放函数指针变量的数组

定义如下:

int (*parr[4])();
//  parr -- 数组名
//  [4] -- 数组元素个数
//  int (*)() -- 数组中元素的类型

6. 转移表

函数指针数组可以用来书写转移表——运用函数指针数组以数组方式去调用里面的函数,从而在某些情况下替代冗长的代码。我们通过计算器的例子来学习一下吧。

计算器的一般实现:

#include <stdio.h>
int add(int a, int b)
{
	return a + b;
}
int sub(int a, int b)
{
	return a - b;
}
int mul(int a, int b)
{
	return a * b;
}
int div(int a, int b)
{
	return a / b;
}
int main()
{
	int x, y;
	int input = 1;
	int ret = 0;
	do
	{
		printf("*************************\n");
		printf("   1:add         2:sub   \n");
		printf("   3:mul         4:div   \n");
		printf(" 0:exit                  \n");
		printf("*************************\n");
		printf("请选择:");
		scanf("%d", &input);
		switch (input)
		{
			case 1:
				printf("输⼊操作数:");
				scanf("%d %d", &x, &y);
				ret = add(x, y);
				printf("ret = %d\n", ret);
				break;
			case 2:
				printf("输⼊操作数:");
				scanf("%d %d", &x, &y);
				ret = sub(x, y);
				printf("ret = %d\n", ret);
				break;
			case 3:
				printf("输⼊操作数:");
				scanf("%d %d", &x, &y);
				ret = mul(x, y);
				printf("ret = %d\n", ret);
				break;
			case 4:
				printf("输⼊操作数:");
				scanf("%d %d", &x, &y);
				ret = div(x, y);
				printf("ret = %d\n", ret);
				break;
			case 0:
				printf("退出程序\n");
				break;
			default:
				printf("选择错误\n");
				break;
		}
	} while (input);
	//计算器的一般实现
	return 0;
}

使用转移表实现计算器:

#include <stdio.h>
int add(int a, int b)
{
	return a + b;
}
int sub(int a, int b)
{
	return a - b;
}
int mul(int a, int b)
{
	return a * b;
}
int div(int a, int b)
{
	return a / b;
}
int main()
{
	int x, y;
	int input = 1;
	int ret = 0;
	int (*arr[5])(int, int) = {0, add, sub, mul, div};//转移表,将函数指针存放进数组中
	do
	{
		printf("*************************\n");
		printf("   1:add         2:sub   \n");
		printf("   3:mul         4:div   \n");
		printf(" 0:exit                  \n");
		printf("*************************\n");
		printf("请选择:");
		scanf("%d", &input);
		if(input >= 1 && input <= 4)
		{
			printf("输入操作数:");
			scanf("%d%d", &x, &y);
			ret = *(arr[input])(x,y);//调用函数指针数组中的元素
			printf("ret=%d\n", ret);
		}
		else if(input == 0)
		{
			printf("程序退出\n");
		}
		else
		{
			printf("输入错误\n");
		}
	} while(input);
	return 0;
}

完。


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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

基于SpringBoot + Vue实现的时装购物管理系统设计与实现+毕业论文+开题报告+答辩PPT

介绍 系统包含用户、管理员两个角色 管理员&#xff1a;首页、个人中心、用户管理、商品分类管理、颜色管理、商品信息管理、商品评价管理、系统管理、订单管理 用户:首页、个人中心、商品评价管理、我的收藏管理、订单管理 前台首页:首页、商品信息、商品资讯、个人中心、后台…

vue 的生命周期--图解

生命周期函数中的this指向是vm 或 组件实例对象。 常用的生命周期钩子&#xff1a; mounted: 发送ajax请求、启动定时器、绑定自定义事件、订阅消息等【初始化操作】。beforeDestroy: 清除定时器、解绑自定义事件、取消订阅消息等【收尾工作】。 关于销毁Vue实例 销毁后借助Vu…

Spring-datasource事务管理-手动请求事务回滚

什么场景下会触发&#xff1f; 在 Spring 中&#xff0c;调用 setRollbackOnly() 方法会将当前事务标记为 rollback-only&#xff0c;表示事务只能回滚&#xff0c;不能提交。这种情况通常发生在以下情景中&#xff1a; 业务逻辑判断&#xff1a;在方法中根据某些业务逻辑的判…

[GFCTF 2021]wordy

用ida查看可知存在大量jmp跳转语令&#xff0c;编写脚本将其改为空指令

MyBatis Dynamic SQL基本使用

MyBatis Dynamic SQL基本使用 一、概念二、特性Hamcrest是什么 三、MyBatis Dynamic SQL 快速入门3.1 环境准备3.2 定义表和列3.3 创建 MyBatis3 映射器3.4 使用 MyBatis3 执行 SQL 四、数据库对象表示4.1 表或视图表示4.2 表别名4.3 列表示 五、Where 子句支持5.1 简单的 wher…

磁盘损坏无法读取:原因、恢复方案与防范之道

在数字化信息爆炸的时代&#xff0c;磁盘作为数据存储的重要载体&#xff0c;承载着无数重要的文件和资料。然而&#xff0c;当磁盘突然损坏&#xff0c;无法读取数据时&#xff0c;我们往往会陷入困境&#xff0c;焦虑不已。面对这种情况&#xff0c;我们该如何应对&#xff1…

晶圆制造之MPW(多项目晶圆)简介

01、MPW是什么&#xff1f; 在半导体行业中&#xff0c;MPW 是 "Multi Project Wafer" 的缩写&#xff0c;中文意思是多项目晶圆。MPW 的主要思想是将使用相同工艺的多个集成电路设计放在同一晶圆片上进行流片&#xff08;即制造&#xff09;。这种方法允许多个设计共…

设计模式-构建者模式

作者持续关注 WPS二次开发专题系列&#xff0c;持续为大家带来更多有价值的WPS二次开发技术细节&#xff0c;如果能够帮助到您&#xff0c;请帮忙来个一键三连&#xff0c;更多问题请联系我&#xff08;QQ:250325397&#xff09; 目录 定义 特点 使用场景 优缺点 (1) 优点 …

26.组件传递Props效验

组件传递Props效验 Vue 组件可以更细致地声明对传入的 props 的校验要求 <template><h3>ComponentA</h3><ComponentB title"Props效验" :userInfo"userInfo"/> </template> <script> import ComponentB from ".…

Linux操作系统·Linux简介

1.世界上第一个完善的网络操作系统 Unix是1969年由美国电话电报公司(AT&T)贝尔实验室的两个工程师所创造的操作系统&#xff0c;它允许计算机同时处理多用户和程序。目前大型政府单位、大型企业、航空公司、金融机构多在使用&#xff0c;价钱昂贵&#xff0c;但性能和稳定性…

【数据结构】99%的人都知道的超好用二叉搜索树

【数据结构】99%的人都知道的超好用二叉搜索树 笔者近期学习了二叉搜索树&#xff0c;与其说是学习了此种数据结构&#xff0c;倒不如说是先在力扣上做了相关题目&#xff0c;而后觉得对其了解甚浅&#xff0c;于是再去找资料…今天就结合力扣题目&#xff0c;向大家介绍一下二…

Spring Cloud 运维篇1——Jenkins CI/CD 持续集成部署

Jenkins 1、Jenkins是什么&#xff1f; Jenkins 是一款开源 CI/CD 软件&#xff0c;用于自动化各种任务&#xff0c;包括构建、测试和部署软件。 Jenkins 支持各种运行方式&#xff0c;可通过系统包、Docker 或者一个独立的 Java 程序。 Jenkins Docker Compose持续集成流…

没有理由不加倍努力

最近su7很火&#xff0c;各隐藏大佬都纷纷从后台来到前台&#xff0c;把整个网红界的网红等级提升了好几个档次。红衣大叔更是借此机会在疯狂地打造自己的网红IP。 千亿大佬都这还般努力&#xff0c;作为平民的自己哪还有不努力的理由。 加倍努力&#xff01;

如何在PostgreSQL中使用pg_stat_statements插件进行SQL性能统计和分析?

文章目录 一、启用pg_stat_statements插件二、查看统计信息三、定期重置统计信息四、注意事项 PostgreSQL中的pg_stat_statements是一个强大的插件&#xff0c;用于追踪执行时间最长的SQL语句。通过它&#xff0c;我们可以获取有关SQL语句执行频率、总执行时间、平均执行时间等…

[创业之路-106] :经济学十大陷阱与核心思想:系统论、社会进化论、周期论、阴阳互转论

目录 前言&#xff1a; 一、流动性陷阱。 二、中等收入陷阱。 三、修昔底德陷阱。 四、塔西佗陷阱。 五、金德尔伯格陷阱。 六、卢梭陷阱。 七、拉美陷阱。 八、阿喀琉斯之踵。 九、布拉德伯里悖论。 十、李约瑟之谜 结论&#xff1a;上述陷阱的…

C++相关概念和易错语法(5)(析构函数、拷贝构造、运算符重载、赋值重载)

上篇文章分享了一些构造函数和析构函数的易错点&#xff0c;这篇文章则将继续分享一些构造函数、拷贝构造函数的易错点。 1.变量声明处赋缺省值 我们已经知道了自动构造函数的初始化规则了。我们可以认为这个初始化规则比较保守&#xff0c;能不修改成员变量的值就不修改&…

实在RPA设计器试用导引

一、产品概述 实在RPA设计器是一款将人工智能(AI)与机器人流程自动化(RPA)深度融合的可视化自动流程编辑器。它通过AI推荐与桌面嵌入式交互&#xff0c;极大简化了RPA的使用难度&#xff0c;让普通业务人员也能轻松使用。实在RPA设计器具备以下核心优势&#xff1a; 兼容性&a…

Redis详解和Spring Data Redis应用

注意事项 如何快速进入命令行窗口什么是配置类 Redis简介 Redis是一个开源的使用ANSI C语言编写的、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库&#xff0c;并提供多种语言的API。它通常被称为数据结构服务器&#xff0c;因为值&#xff08;value&#xff09…

数电期末复习(二)逻辑代数基础

这里写目录标题 2.1 二值逻辑变量与基本逻辑运算2.1.1 与运算2.1.2 或运算2.1.3 非运算2.1.4 常用复合逻辑运算 2.2 逻辑函数的建立及其表示方法2.2.1 真值表表示2.2.2 逻辑函数表达式表示2.2.3 逻辑图表示方法2.2.4 波形图表示方法 2.3 逻辑代数2.3.1 逻辑代数的基本定律和恒等…