第十一讲:指针(3)

第十一讲:指针(3)

  • 1.字符指针变量
    • 1.1存储一个字符
    • 1.2存储一个字符串
    • 1.3一个有趣的面试题
  • 2.数组指针变量
    • 2.1什么是数组指针变量
    • 2.2数组指针变量的初始化
  • 3.二维数组传参的本质
  • 4.函数指针变量
    • 4.1介绍函数指针变量
    • 4.2 两段有趣的代码
      • 4.2.1代码1
      • 4.2.2代码2
    • 4.3typedef关键字
  • 5.函数指针数组
  • 6.转移表
    • 6.1一般写法
    • 6.2函数指针数组写法

1.字符指针变量

1.1存储一个字符

我们当然可以使用指针来存放一个字符变量,使用方法如下:
在这里插入图片描述

1.2存储一个字符串

那么如果我们用指针存放字符串会发生什么呢?

结论:
1.指针存放字符串时存放的时一个字符串首元素的地址
2.直接使用指针存放字符串时,这个字符串被称为常量字符串,此时字符串的值不能被改变


int main()
{char* pa = "abcdef";   //此时abcdef就是一个常量字符串//pa中存储的是首元素a的地址,证明如下:printf("%c\n", *pa);  //此时打印出的是a//*pa = 'c';  如果想要改变pa的值,程序会崩溃,但是并不会报错//所以我们不如将上面的代码表示如下const char* pb = "abcd";//*pb = 0;   此时如果再想改变pb的值,程序就会报错return 0;
}

1.3一个有趣的面试题

在这里插入图片描述
这个面试题的结果为:
在这里插入图片描述
原因如下:
str3和str4指向的是同一个常量字符串,常量字符串会存储再内存中的常量区,它的地址是确定的,而使用str1和str2初始化数组时,就会开辟不同的内存块

2.数组指针变量

2.1什么是数组指针变量

类比于
整形指针变量(int * pa),存放的时整形变量的地址,能够指向整形数据
浮点型指针变量(float * pb),存放的是浮点型变量的地址,指向浮点型数据
那么数组指针变量应该是:
存放的是数组的地址,能够指向数组的指针变量

指针变量的表示方法如下:
在这里插入图片描述
pa先和*进行结合,说明pa是一个指针变量,指向的是一个大小为5的数组。所以pa是一个指针,指向一个数组,称为数组指针

2.2数组指针变量的初始化

在这里插入图片描述
如果要存放个数组的地址,就得存放在数组指针变量中
数组指针解析:
在这里插入图片描述

3.二维数组传参的本质

结论:
二维数组传参,传入的是二维数组中首个一维数组的地址

我们用代码和图像来进一步理解二维数组传参的本质:


//二维数组传参的本质 (打印二维数组的每一个元素)
Print(int(*pa) [3], int r, int c)  //二维数组传参的本质是传入了首个一维数组的地址,而首个一维数组的类型是int [3],
{                                   //所以接受的指针类型是int (*)[3]的类型for (int i = 0; i < r; i++){for (int j = 0; j < c; j++){printf("%d ", (*(pa + i))[j]); //这里的(*(pa + i))[j]相当于*(*(pa+i)+j)相当于pa[i][j]}printf("\n");}
}int main()
{int arr[2][3] = { {1,2,3}, {2,3,4} };Print(arr, 2, 3);return 0;
}

在这里插入图片描述
在这里插入图片描述
当然,二维数组进行传参时,也可以用二维数组来接受:

Print(int arr[2][3], int r, int c)

4.函数指针变量

4.1介绍函数指针变量

类比于数组指针变量(用来存放数组的地址),函数指针变量是用来存放函数的地址的,对于函数的地址的理解:
在这里插入图片描述
我们可以看出来,函数名其实就是函数的地址,Add == &Add,为了将函数的地址存下来,就要使用函数指针变量,使用方法如下:


//函数指针的使用
int Add(int x, int y)
{return x + y;
}int main()
{int a = 2;int b = 3;//(int x, int y)中x和y写不写都可以int (*Pa)(int, int) = Add; //将Add函数的地址存储在函数指针变量中,Add和&Add等价printf("%d\n", Pa(a, b));  //在使用时,pa(a,b)和(*pa)(a,b)等价return 0;
}

函数指针类型解析:
在这里插入图片描述

4.2 两段有趣的代码

4.2.1代码1

(*(void (*)())0)();

对于上述代码的解释如下:
在这里插入图片描述

4.2.2代码2

void (*signal(int , void(*)(int)))(int);

在这里插入图片描述

4.3typedef关键字

typedef是用来类型重命名的,可以将复杂的类型简单化,具体使用方法如下:


//typedef关键字
typedef unsigned int uint;    //1.使用关键字定义普通类型的名称
typedef int* pa;              //2.使用关键字定义指针类型的名称
typedef int(*ppa)[4];         //3.使用关键字定义数组指针的名称
typedef int(*pppa)(int, int); //4.使用关键字定义函数指针变量名称int Add(int x, int y)
{return x + y;
}int main()
{uint a = 3;  //关键字的使用printf("%d\n", a);pa p_a = &a;*p_a = 4;printf("%d\n", a);int arr[4] = { 1,2,3,4 };ppa p_arr = &arr;for (int i = 0; i < 4; i++){printf("%d ", *((*p_arr) + i));}printf("\n");pppa p_add = Add;int ret = p_add(2, 3);printf("%d", ret);return 0;
}

那么此时我们就可以简化代码2,将其转换成:

void (*signal(int , void(*)(int)))(int);typedef void(*pfun_t)(int);
pfun_t signal(int, pfun_t);

5.函数指针数组

函数指针数组是存放函数指针的数组,那么函数指针数组如何使用呢?


//函数指针数组的使用
int Add(int x, int y)
{return x + y;
}int main()
{int (*pa[3])(int, int) = { Add };int ret = pa[0](2, 3);printf("%d", ret);return 0;
}

6.转移表

函数指针的用途:转移表
举例:计算器的一般实现

6.1一般写法


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

显然,这个方法比较啰嗦,不好用

6.2函数指针数组写法


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 (*pa[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 <= 5){printf("请输入两个操作数:");scanf("%d %d", &x, &y);int ret = pa[input](x, y);printf("%d\n", ret);}else if (input == 0){printf("退出程序!");break;}elseprintf("输入的值非法,重新输入:");} while (input);return 0;
}

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

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

相关文章

TP5000是一款开关降压型单节锰锂电池/磷酸铁锂电池充电管理芯片IC

一般概述 TP5000是一款开关降压型单节锰锂电池/磷酸铁锂电池充电管理芯片。其QFN16超小型封装与简单的外围电路&#xff0c;使得TP5000非常适用于便携式设备的大电流充电管理应用。同时&#xff0c;TP5000内置输入过流、欠压保护、芯片过温保护、短路保护、电池温度监控…

videosapi、强大的微信管理api

接口地址&#xff1a;videosapi.com 同步个人收藏夹 {"appId": "{{appid}}","syncKey": "" } 出参 {"ret": 200,"msg": "操作成功","data": {"syncKey": "CAESCAgBEJyi9e4…

设计模式1——初步认识篇

设计模式1——初步认识篇 一、先让我们浅聊一下面向对象和设计模式。 说起设计模式&#xff0c;我第一次听到它&#xff0c;是在学习面向对象的时候。那么什么是面向对象&#xff0c;什么是设计模式&#xff0c;而且设计模式和面向对象又有什么关系呢&#xff1f; 1、什么是面…

G.AB路线【蓝桥杯】/bfs+可重复走

AB路线 bfs可重复走 思路&#xff1a;本题和传统的bfs题目不同&#xff0c;本题为了满足题目先走K个A再走K个B&#xff0c;可能需要重复走某个格子才能继续走下去&#xff0c;故vis数组可以多开一维&#xff0c;vis[x][y][z]表示第z次走到x行y列这种情况是否出现过 A A A B B …

通用型产品发布解决方案(后端环境搭建)

文章目录 后端renren脚手架配置1.解压后放到项目目录下2.新建商品模块1.创建一个新模块 sunliving-commodity2.删除两个不必要的文件3.pom.xml 引入依赖 3.maven进行聚合管理1.将刚才配置的pom.xml文件复制到父项目下并进行修改2.手动将这个pom.xml加入项目&#xff08;如果右下…

vs2019 里 C++ 20规范的 string 类的源码注释

&#xff08;1&#xff09;读源码&#xff0c;可以让我们更好的使用这个类&#xff0c;掌握这个类&#xff0c;知道咱们使用了库代码以后&#xff0c;程序大致具体是怎么执行的。而不用担心程序出不知名的意外的问题。也便于随后的代码调试。 string 类实际是 库中 basic_strin…

R2S+ZeroTier+Trilium

软路由使用ZeroTier搭建远程笔记 软路由使用ZeroTier搭建远程笔记 环境部署 安装ZeroTier安装trilium 环境 软路由硬件&#xff1a;友善 Nanopo R2S软路由系统&#xff1a;OpenWrt&#xff0c;使用第三方固件nanopi-openwrt。内网穿透&#xff1a;ZeroTier。远程笔记&…

数据分析:基于sparcc的co-occurrence网络

介绍 Sparcc是基于16s或metagenomics数据等计算组成数据之间关联关系的算法。通常使用count matrix数据。 安装Sparcc软件 git clone gitgithub.com:JCSzamosi/SparCC3.git export PATH/path/SparCC3:$PATHwhich SparCC.py导入数据 注&#xff1a;使用rarefy抽平的count ma…

自动驾驶学习2-毫米波雷达

1、简介 1.1 频段 毫米波波长短、频段宽,比较容易实现窄波束,雷达分辨率高,不易受干扰。波长介于1~10mm的电磁波,频率大致范围是30GHz~300GHz 毫米波雷达是测量被测物体相对距离、相对速度、方位的高精度传感器。 车载毫米波雷达主要有24GHz、60GHz、77GHz、79GHz四个频段。 …

Redis数据结构-Dict

1.3 Redis数据结构-Dict 我们知道Redis是一个键值型&#xff08;Key-Value Pair&#xff09;的数据库&#xff0c;我们可以根据键实现快速的增删改查。而键与值的映射关系正是通过Dict来实现的。 Dict由三部分组成&#xff0c;分别是&#xff1a;哈希表&#xff08;DictHashTa…

【InternLM】书生·浦语大模型报名地址

【InternLM】书生浦语大模型报名地址&#xff1a; 扫码报名

药物代谢动力学学习笔记

一、基本概念 二、经典房室模型 三、非线性药物代谢动力学 四、非房室模型 五、药代动力学与药效动力学 六、生物等效性评价 七、生物样品分析方法 基本概念 生物样品&#xff1a;生物机体的全血、血浆、血清、粪便、尿液或其他组织的样品 特异性&#xff0c;specificity&…