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

news2025/1/16 18:04:49

目录

  • 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;
}

完。


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

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

相关文章

融合算法:引力融合

问题&#xff1a; 有一批数据&#xff0c;如&#xff1a; [ 1, 2, 3, 45, 46, 55, 89, 101 ]想把它分成3块&#xff0c;如&#xff1a; [1, 2, 3] [45, 46, 55] [89, 101]算法&#xff1a; 参考万有引力公式&#xff0c;想象坐标轴上这8个点的分布&#xff1a; 每一个点都会…

偏执型人格的症状和起因,偏执型人格测试和应对方法

偏执型人格&#xff0c;主打一个敏感多疑&#xff0c;过分的固执&#xff0c;过度的警觉。无论别人怎么解析&#xff0c;都难以说服。偏执型人格常常毫无根据的怀疑他人的忠诚&#xff08;动机&#xff09;&#xff0c;曲解别人的言语&#xff08;或行为&#xff09;&#xff0…

MySQL--表的操作

目录 创建表 查看表结构 修改表 新增列 修改列类型 修改列名 修改表名&#xff1a; 删除列 删除表 创建表 语法&#xff1a; CREATE TABLE table_name ( field1 datatype, field2 datatype, field3 datatype ) character set 字符集 collate 校验规则 engine 存储引…

数据结构 - 顺序表

一. 线性表的概念 线性表&#xff08;linear list&#xff09;是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构&#xff0c;常见的线性表&#xff1a;顺序表、链表、栈、队列、字符串... 线性表在逻辑上是线性结构&#xff0c;也就说是连续的…

分类算法——决策树(五)

认识决策树 决策树思想的来源非常朴素&#xff0c;程序设计中的条件分支结构就是if-else结构&#xff0c;最早的决策树就是利用这类结构分割数据的一种分类学习方法。 决策树分类原理详解 为了更好理解决策树具体怎么分类的&#xff0c;通过一个问题例子&#xff1a; 问题…

每日OJ题_BFS解决拓扑排序③_力扣LCR 114. 火星词典

目录 力扣LCR 114. 火星词典 解析代码 力扣LCR 114. 火星词典 LCR 114. 火星词典 难度 困难 现有一种使用英语字母的外星文语言&#xff0c;这门语言的字母顺序与英语顺序不同。 给定一个字符串列表 words &#xff0c;作为这门语言的词典&#xff0c;words 中的字符串已…

[Java EE] 多线程(三):线程安全问题(上)

1. 线程安全 1.1 线程安全的概念 如果多线程环境下代码运行的结果不符合我们的预期,则我们说存在线程安全问题,即程序存在bug,反之,不存在线程安全问题. 1.2 线程不安全的原因 我们下面举出一个线程不安全的例子:我们想要在两个线程中对count进行操作 public class Demo9 …

RC电路延时时间常数在线计算器

RC电路延时时间常数在线计算器: https://www.838dz.com/calculator/1888.html 急用时&#xff0c;找不到。

后端通过@jsonformat格式化数据转发,前端无法正确显示

后端发送给前端的updatatime是有格式的 后端接收的数据没有任何变化&#xff0c;前端代码也很正常 显示时间也乱码 原因应该是某个注释和jsonformat冲突了&#xff0c;所幸就不用jesonformat 用手动配置的消息转换器 // 消息转换器&#xff0c;后端返回给前端数据格式化Overri…

用斐波那契数列感受算法的神奇(21亿耗时0.02毫秒)

目录 一、回顾斐波那契数列 二、简单递归方法 &#xff08;一&#xff09;解决思路 &#xff08;二&#xff09;代码展示 &#xff08;三&#xff09;性能分析 三、采用递归HashMap缓存 &#xff08;一&#xff09;解决思路 &#xff08;二&#xff09;代码展示 &…

C++入门----内联函数auto范围fornullptr指针

1.内联函数 顾名思义&#xff0c;内联函数也是函数的一种&#xff0c;我们在C语言的学习过程里面知道了函数和宏之间的区别和各自的优缺点&#xff1b; 函数的使用需要建立栈帧&#xff0c;宏的使用需要考虑各种符号的优先级问题&#xff0c;很容易出错&#xff0c;因为宏在使…

新恒盛110kV变电站智能辅助系统综合监控平台+道巡检机器人

江苏晋控装备新恒盛化工有限公司是晋能控股装备制造集团有限公司绝对控股的化工企业&#xff0c;公司位于江苏省新沂市。新恒盛公司40•60搬迁项目在江苏省新沂市经济开发区化工产业集聚区苏化片区建设&#xff0c;总投资为56.64亿元&#xff0c;该项目是晋能控股装备制造集团重…

Spring - 5 ( 8000 字 Spring 入门级教程 )

一&#xff1a;Spring IoC&DI 1.1 方法注解 Bean 类注解是添加到某个类上的&#xff0c; 但是存在两个问题: 使用外部包里的类, 没办法添加类注解⼀个类, 需要多个对象, ⽐如多个数据源 这种场景, 我们就需要使用方法注解 Bean 我们先来看方法注解如何使用: public c…

YOLOv3没有比这详细的了吧

YOLOv3&#xff1a;目标检测基于YOLOv2的改进 在目标检测领域&#xff0c;YOLO&#xff08;You Only Look Once&#xff09;系列以其出色的性能和速度而闻名。YOLOv3作为该系列的第三个版本&#xff0c;不仅继承了前身YOLOv2的优势&#xff0c;还在多个方面进行了创新和改进。…

Linux中的高级IO函数(一)pipe socketpair dup

Linux提供了很多高级的I/O函数。它们并不像Linux基础I/O函数&#xff08;比如open和read&#xff09;那么常用&#xff08;编写内核模块时一般要实现这些I/O函数&#xff09;&#xff0c;但在特定的条件下却表现出优秀的性能。这些函数大致分为三类&#xff1a; 用于创建文件描…

HarmonyOS开发案例:【闹钟】

介绍 使用后台代理提醒&#xff0c;实现一个简易闹钟。要求完成以下功能&#xff1a; 展示指针表盘或数字时间。添加、修改和删除闹钟。展示闹钟列表&#xff0c;并可打开和关闭单个闹钟。闹钟到设定的时间后弹出提醒。将闹钟的定时数据保存到轻量级数据库。 相关概念 [Canva…

数据结构入门——排序(代码实现)(下)

int GetMidi(int* a, int left, int right) {int mid (left right) / 2;// left mid rightif (a[left] < a[mid]){if (a[mid] < a[right]){return mid;}else if (a[left] > a[right]) // mid是最大值{return left;}else{return right;}}else // a[left] > a[mid…

MySQL-----多表查询(一)

目录 一.多表关系&#xff1a; 1.1 一对多(多对一)&#xff1a; 1.2 多对多: 1.3 一对一: 二.多表查询概述&#xff1a; 三.连接查询&#xff1a; 3.1内连接&#xff1a; 3.2外连接&#xff1a; 3.3自连接查询&#xff1a; 3.4联合查询&#xff1a; 一.多表关系&…

测试的分类(3)

目录 按照测试阶段测试 系统测试 冒烟测试和回归测试的区别 验收测试 单元测试, 集成测试, 系统测试, 回归测试之间的关系 是否按手工进行测试 手工测试 自动化测试 自动化测试和手工测试的优缺点 自动化测试优点 自动化测试缺点 手工测试优点 手工测试缺点 按照…

鸿蒙HarmonyOS应用 - ArkUI组件

ArkUI组件 基础组件 Image 声明Image组件并设置图片源 网络权限&#xff1a;ohos.permission.INTERNET Image(scr: string | PixelMap | Resource)// 1. string&#xff1a;用于加载网络图片&#xff0c;需要申请网络权限 Image("https://xxx.png")// 2. PixelMap…