c语言-浅谈指针(3)
文章目录
- 1.字符指针变量
- 常见的字符指针初始化
- 另一种字符指针初始化
- 例:
- 2.数组指针变量
- 什么是数组指针变量
- 数组指针变量创建
- 数组指针变量初始化
- 例(二维数组传参的本质)
- 3.函数指针变量
- 什么是函数指针变量呢?
- 函数指针变量创建
- 函数指针使用
- typedef关键字
- 4.函数指针数组
- 概念
- 创建
- 通过函数指针数组实现计算器
1.字符指针变量
常见的字符指针初始化
字符指针 char *
常见的初始化:
int main() {char a = 'w';char* p = &a;return 0;
}
跟 int *类型的初始化一样,下面介绍另一种字符指针的初始化
另一种字符指针初始化
这种是将一个字符串直接赋给字符指针变量
如:
char* p = "abcdefg";
那么它能不能像一般的字符数组那样直接打印出来呢,我们通过代码看看:
int main() {char a[] = "abcdefg";char* p = "abcdefg";printf("%s", a);//打印字符数组printf("%s", *p);//打印指针初始化出来的字符串return 0;
}
运行结果:

由结果看出:只有字符数组打印出来了,而字符指针没打印出来,这是为什么呢? 其实是因为这样初始化只是把字符串的首个字符的地址传给字符指针变量 p ,这样的话 *p=a。 在这里%s是打印字符串的,所以打印不了一个字符。
那我们用打印一个字符的 %c 试试
int main() {char a[] = "abcdefg";char* p = "abcdefg";printf("%s\n", a);printf("%c", *p);return 0;
}
运行结果

字符指针只打印一个首字符,验证了首个字符的地址传给字符指针变量 p
那么我们可以用循环的方式将这个字符串打印出来
int main() {char* p = "abcdefg";int s = strlen(p);//求字符长度
for(int i=0;i<s;i++)printf("%c", *(p+i));//将字符一一打印出来return 0;
}
运行结果

例:
这是《剑指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;}
答案:

分析:
1.这道题其实就是比较首地址 2这⾥str3和str4指向的是⼀个同⼀个常量字符串。C/C++会把常量字符串存储到单独的⼀个内存区域,当⼏个指针指向同⼀个字符串的时候,他们实际会指向同⼀块内存。但是⽤相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4相同。
2.数组指针变量
数组指针是指针不是数组
什么是数组指针变量
我们通过其他类型的指针进行类比一下 整形指针变量: int * pint; 存放的是整形变量的地址,能够指向整形数据的指针。 浮点型指针变量: float * pf; 存放浮点型变量的地址,能够指向浮点型数据的指针。 那么数组指针变量就是存放数组地址,指向数组的指针变量吧
数组指针变量创建
创建:int (*p)[10]
解释:p先和*结合,说明p是⼀个指针变量变量,然后指着指向的是⼀个⼤⼩为10个整型的数组。所以p是⼀个指针,指向⼀个数组,叫数组指针。
注意:[ ]的优先级要⾼于 * 号的,所以必须加上()来保证p先和 * 结合。
数组指针变量初始化
数组指针是用来存放数组的指针的,那么我们将数组取地址在赋给数组指针变量即可
如:
int arr[10] = { 1,2,3,4,5,7,8,9,0.6 };
int(*p)[10] = &arr;
解释:

图解(一维数组)

例(二维数组传参的本质)
在数组里面的二维数组可以看成一个一维数组,只不过是每个元素就是一个数组而已,所以,根据数组名是数组⾸元素的地址这个规则,⼆维数组的数组名表⽰的就是第⼀⾏的地址,是⼀维数组的地址。

1.通过二维数组接收参数
这是一般的方法
void qin(int arr[2][5] ){for (int i = 0; i < 2; i++) {for (int j = 0; j < 5; j++)printf("%d ", arr[i][j]);}printf("\n");
}
int main() {int arr[2][5] = { 1,2,3,4,5,6,7,8,9,0 };qin(arr);return 0;
}
2.通过数组指针来了解二维数组的本质
void qin(int(*p)[5]) {//*(p+1)==arr[0]+jfor (int i = 0; i < 2; i++) {for (int j = 0; j < 5; j++)printf("%d ", (*(p + i))[j]);// //(*p)==*(p+i),当i 加1时相当于跳过二维数组中的一个一维数组,//当然跟据前面的知识也可以写成这样 *(*(p+i)+j)
}printf("\n");
}
int main() {int arr[2][5] = { 1,2,3,4,5,6,7,8,9,0 };qin(arr);//在前面指针内容里说过,数组名就是首元素的地址,//那么我们这里把二维数组当成一维数组,那么首元素不就是一整个一维数组的地址吗return 0;
}
图解:

总结:⼆维数组传参,形参的部分可以写成数组,也可以写成指针形式。
3.函数指针变量
什么是函数指针变量呢?
数组指针是用来存数组的地址的那么函数指针呢?
其实函数指针也是用来存地址的,存的是函数的地址,这里就有很多人有疑问了,函数有地址?
那么我们来做个测试:
int add(int x, int y) {return x + y;
}
int main() {printf("%p", &add);return 0;
}
运行结果:

很明显函数是有地址的
这里还有一个注意的是函数名和&函数名都是代表函数的地址,如:add==&add
我们来测试一下吧
int add(int x, int y) {return x + y;
}
int main() {printf("&add=%p\n", &add);printf("add=%p", add);return 0;
}
运行结果:

耶,是一样的
函数指针变量创建
创建:int (*p) (int, int)

函数指针使用
我们用函数指针来实现一下加法
int add(int x, int y) {return x + y;
}
int main() {int (*p)(int, int) = add;int a = 3, b = 9;printf("%d\n", p(a, b));//这里不用 * 也可以因为函数调用的时候本身就是用地址去调用printf("%d\n", (*p)(a, b));return 0;
}
运行结果:

typedef关键字
typedef 是⽤来类型重命名的,可以将复杂的类型,简单化。
如:
typedef unsigned int uint;
//将unsigned int 重命名为uint
那么如何来重命名指针类型呢
typedef int(*)(int ,int) te //错误
typedef int(*te)(int, int) //正确命名,将重命名放到原来放函数指针名称的位置
typedef (*te)[10]//数组指针重命名
4.函数指针数组
概念
把函数的地址存到⼀个数组中,那这个数组就叫函数指针数组(就是将函数指针放到一个数字里)
创建
int (*) (int,int)//函数指针类型
int (*p[10])(int ,int)//函数指针数组--p先和[]结合形成数组,再和函数指针类型结合形成函数指针数组
通过函数指针数组实现计算器
我们这里要实现一个简单的计算器,分别有加法、减法、乘法、除法,那么我们就要实现四个函数来实现,然后还要一个选择来选择实行什么计算,最后还需要一个菜单就完成了
我们先用一般的方法来实现计算器:
#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;
}
使⽤函数指针数组的实现:
我们将这4个函数都放到一个函数指针数组里,当我们需要实行什么计算就通过什么下标来访问这个函数
#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(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //储存函数地址,这里的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);//输入下标if ((input <= 4 && input >= 1)){printf( "输⼊操作数:" );scanf( "%d %d", &x, &y);ret = (*p[input])(x, y);printf( "ret = %d\n", ret);}else if(input == 0){printf("退出计算器\n");}else{printf( "输⼊有误\n" ); }
以上就是我的分享
谢谢大家的观看!
相关文章:
c语言-浅谈指针(3)
文章目录 1.字符指针变量常见的字符指针初始化另一种字符指针初始化例: 2.数组指针变量什么是数组指针变量数组指针变量创建数组指针变量初始化例(二维数组传参的本质) 3.函数指针变量什么是函数指针变量呢?函数指针变量创建函数指…...
从服务器端获取人脸数据,在本地检测特征,并将特征发送给服务器
目录 1.定义函数get_database_process: 2.定义函数features_construct: 3.定义函数send_features_data: 4. 定义函数database_features_construct: 5. main 函数 1.定义函数get_database_process: …...
ARDUINO UNO 12颗LED超酷流水灯效果
效果代码: #define t 30 #define t1 20 #define t2 100 #define t3 50 void setup() { // set up pins 2 to 13 as outputs for (int i 2; i < 13; i) { pinMode(i, OUTPUT); } } /Effect 1 void loop() { effect_1(); effect_1(); effect_…...
Linux下查看pytorch运行时真正调用的cuda版本
一般情况我们会安装使用多个cuda版本。而且pytorch在安装时也会自动安装一个对应的版本。 正确查看方式: 想要查看 Pytorch 实际使用的运行时的 cuda 目录,可以直接输出 cpp_extension.py 中的 CUDA_HOME 变量。 import torch import torch.utils imp…...
分享mfc140u.dll丢失的解决方法,针对原因解决mfc140u.dll丢失的问题
作为电脑小白,如果电脑中出现了mfc140u.dll丢失的问题,肯定会比较的慌乱。但是出现mfc140u.dll丢失的问题,其实也有很简单的办法,所以大家不用慌张,接下来就教大家解决办法,能够有效的解决mfc140u.dll丢失的…...
torch_cluster、torch_scatter、torch_sparse三个包的安装
涉及到下面几个包安装的时候经常会出现问题,这里我使用先下载然后再安装的办法: pip install torch_cluster pip install torch_scatter pip install torch_sparse 1、选择你对应的torch版本:https://data.pyg.org/whl/ 2、点进去然后&…...
软件安利——火绒安全
近年来,以优化、驱动、管理为目标所打造的软件屡见不鲜,大同小异的电脑管家相继走入了公众的视野。然而,在这日益急功近利的社会氛围驱动之下,真正坚持初心、优先考虑用户体验的电脑管家逐渐湮没在了浪潮之中。无论是鲁大师&#…...
Induced AI:一个专门为自动化任务而设计的AI原生浏览器RPA平台
内容来源:xiaohuggg Induced AI:一个专门为自动化任务而设计的AI原生浏览器RPA平台 刚刚获得OpenAI CEOsama的个人投资! 它能够模拟人类浏览网页的行为,自动化地浏览网页,搜集关键信息,并对这些信息进行…...
vue3中使用reactive定义的变量响应式丢失问题(大坑!!!)
前言 在Vue 3中,可以使用reactive函数将普通JavaScript对象转换为响应式对象,这样当对象的属性发生变化时,就会自动更新相应的UI。 但是请注意以下情况可能会丢失数据的响应式: 响应式丢失的情况: 1、对使用reactiv…...
Windows Server 2012 R2系统服务器远程桌面服务多用户登录配置分享
Windows Server 2012系统在没有安装远程多界面的情况下,最多只能同时运行2个远程桌面,如果是有多个技术员、合伙人同时操作或是像游戏开发需要用到多界面,但是没有安装就很不方便,今天飞飞来和你们分享Windows server 2012R2系统远…...
mysql之搭建MHA架构实现高可用
1、定义 全称是masterhigh avaliabulity。基于主库的高可用环境下可以实现主从复制及故障切换(基于主从复制才能故障切换) MHA最少要求一主两从,半同步复制模式 2、作用 解决mysql的单点故障问题。一旦主库崩溃,MHA可以在0-30…...
Databend 与海外某电信签约:共创海外电信数据仓库新纪元
为什么选择 Databend 海外某电信面临的主要挑战是随着业务量的增加,传统的 Clickhouse Hive 方案在数据存储和处理上开始显露不足。 原来的大数据分析采用的 Clickhouse Hive 方案进行离线的实时报表。但随着业务量的上升后,Hive的数据存储压力变大&…...
scala解析命令行参数
如何用scala解析命令行参数: 首先,需要在项目中添加Apache Commons CLI库的依赖。可以在build.sbt文件中添加如下行: libraryDependencies "commons-cli" % "commons-cli" % "1.4" import org.apache.comm…...
盘点60个Python各行各业管理系统源码Python爱好者不容错过
盘点60个Python各行各业管理系统源码Python爱好者不容错过 学习知识费力气,收集整理更不易。 知识付费甚欢喜,为咱码农谋福利。 源码下载链接:https://pan.baidu.com/s/1VdAFp4P0mtWmsA158oC-aA?pwd8888 提取码:8888 项目名…...
SpringSecurity6 | 自动配置(下)
✅作者简介:大家好,我是Leo,热爱Java后端开发者,一个想要与大家共同进步的男人😉😉 🍎个人主页:Leo的博客 💞当前专栏: Java从入门到精通 ✨特色专栏…...
6、传统CV之均值滤波
在前5节,从最基础的像素开始了介绍,并且着重介绍了像素局部性、RGB图片和YUV图片以及通道的概念。 其实写那么多,很多细节知识也不用都学会,只需要知道计算机在处理图片时,看到的都是一堆像素,而这一堆像素,都是以数据点的形式存放在计算机中的。 为了更好的展示图像和…...
快速搭建本地的chatgpt
快速搭建本地的chatgpt 参考:一篇文章教你使用Docker本地化部署Chatgpt(非api,速度非常快!!!)及裸连GPT的方式(告别镜像GPT)-CSDN博客 前提是linux下 已安装docker 命…...
分布式下多节点WebSocket消息收发
1、使用场景 2、疑问 第一次发送请求后,通过N1,W2,到达service2,建立websocket连接。 1、接下来发送的消息,通过Ngixn后和网关gateway后还能落在service2上面吗? 如果不能落在service2上,需要怎…...
LeetCode算法题解(动态规划)|LeetCode509. 斐波那契数、LeetCode70. 爬楼梯、LeetCode746. 使用最小花费爬楼梯
一、LeetCode509. 斐波那契数 题目链接:509. 斐波那契数 题目描述: 斐波那契数 (通常用 F(n) 表示)形成的序列称为 斐波那契数列 。该数列由 0 和 1 开始,后面的每一项数字都是前面两项数字的和。也就是:…...
【图像处理】:Otsu算法最大类间方差法(大津算法:附源码)
这里写自定义目录标题 数学原理算法评价参考链接 数学原理 以灰度图像为例,对于图像MN大小的矩阵,即图像中的像素,每一个值即为像素值,其中灰度图像像素值在(0~255)之间。 主要实现前景(即目标)和背景的分割: 主要公式…...
RestClient
什么是RestClient RestClient 是 Elasticsearch 官方提供的 Java 低级 REST 客户端,它允许HTTP与Elasticsearch 集群通信,而无需处理 JSON 序列化/反序列化等底层细节。它是 Elasticsearch Java API 客户端的基础。 RestClient 主要特点 轻量级ÿ…...
智慧工地云平台源码,基于微服务架构+Java+Spring Cloud +UniApp +MySql
智慧工地管理云平台系统,智慧工地全套源码,java版智慧工地源码,支持PC端、大屏端、移动端。 智慧工地聚焦建筑行业的市场需求,提供“平台网络终端”的整体解决方案,提供劳务管理、视频管理、智能监测、绿色施工、安全管…...
跨链模式:多链互操作架构与性能扩展方案
跨链模式:多链互操作架构与性能扩展方案 ——构建下一代区块链互联网的技术基石 一、跨链架构的核心范式演进 1. 分层协议栈:模块化解耦设计 现代跨链系统采用分层协议栈实现灵活扩展(H2Cross架构): 适配层…...
(转)什么是DockerCompose?它有什么作用?
一、什么是DockerCompose? DockerCompose可以基于Compose文件帮我们快速的部署分布式应用,而无需手动一个个创建和运行容器。 Compose文件是一个文本文件,通过指令定义集群中的每个容器如何运行。 DockerCompose就是把DockerFile转换成指令去运行。 …...
Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理
引言 Bitmap(位图)是Android应用内存占用的“头号杀手”。一张1080P(1920x1080)的图片以ARGB_8888格式加载时,内存占用高达8MB(192010804字节)。据统计,超过60%的应用OOM崩溃与Bitm…...
C# 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...
JavaScript 数据类型详解
JavaScript 数据类型详解 JavaScript 数据类型分为 原始类型(Primitive) 和 对象类型(Object) 两大类,共 8 种(ES11): 一、原始类型(7种) 1. undefined 定…...
MySQL JOIN 表过多的优化思路
当 MySQL 查询涉及大量表 JOIN 时,性能会显著下降。以下是优化思路和简易实现方法: 一、核心优化思路 减少 JOIN 数量 数据冗余:添加必要的冗余字段(如订单表直接存储用户名)合并表:将频繁关联的小表合并成…...
C# 表达式和运算符(求值顺序)
求值顺序 表达式可以由许多嵌套的子表达式构成。子表达式的求值顺序可以使表达式的最终值发生 变化。 例如,已知表达式3*52,依照子表达式的求值顺序,有两种可能的结果,如图9-3所示。 如果乘法先执行,结果是17。如果5…...
手机平板能效生态设计指令EU 2023/1670标准解读
手机平板能效生态设计指令EU 2023/1670标准解读 以下是针对欧盟《手机和平板电脑生态设计法规》(EU) 2023/1670 的核心解读,综合法规核心要求、最新修正及企业合规要点: 一、法规背景与目标 生效与强制时间 发布于2023年8月31日(OJ公报&…...
