C语言指针常见问题汇总
我们在学C语言时,指针是我们最头疼的问题之一,针对C语言指针,博主根据自己的实际学到的知识以及开发经验,总结了以下使用C语言指针时常见问题。
1、指针做函数参数
学习函数的时候,讲了函数的参数都是值拷贝,在函数里面改变形参的值,实参并不会发生改变。
如果想要通过形参改变实参的值,就需要传入指针了。
注意:虽然指针能在函数里面改变实参的值,但是函数传参还是值拷贝。不过指针虽然是值拷贝,但是却指向的同一片内存空间。
2、指针做函数返回值
返回指针的函数,也叫作指针函数。
和普通函数一样,只是返回值类型不同而已,先看一下下面这个函数,非常熟悉对不!
int fun(int x,int y);
接下来看另外一个函数声明
int* fun(int x,int y);
这样一对比,发现所谓的指针函数也没什么特别的。
注意:
-
不要返回临时变量的地址
-
可以返回动态申请的空间的地址
-
可以返回静态变量和全局变量的地址
3、函数指针
如果在程序中定义了一个函数,那么在运行时系统就会为这个函数代码分配一段存储空间,这段存储空间的首地址称为这个函数的地址。而且函数名表示的就是这个地址。既然是地址我们就可以定义一个指针变量来存放,这个指针变量就叫作函数指针变量,简称函数指针。
(1)函数指针定义
函数返回值类型 (* 指针变量名) (函数参数列表);
-
“函数返回值类型”表示该指针变量所指向函数的 返回值类型;
-
“函数参数列表”表示该指针变量所指向函数的参数列表。
那么怎么判断一个指针变量是指向变量的指针,还是指向函数的指针变量呢?
-
看变量名的后面有没有带有形参类型的圆括号,如果有就是指向函数的指针变量,即函数指针,如果没有就是指向变量的指针变量。
-
函数指针没有++和 --运算
(2)函数指针使用
定义一个实现两个数相加的函数。
int add(int a,int b)
{return a+b;
}
int main()
{int (*pfun)(int,int) = add;int res = pfun(5,3);printf("res:%d\n",res);return 0;
}
在给函数指针pfun赋值时,可以直接用add赋值,也可以用&add赋值,效果是一样的。
在使用函数指针时,同样也有两种方式,1,pfun(5,3); 2,(*pfun)(5,3)
(3)案例
计算器
用函数指针实现一个简单的计算器,支持+、-、*、/、%
//plus sub multi divide mod //加 减 乘 除 取余
当功能太多时,switch语句太长,因此不是一种好的编程风格。好的设计理念应该是把具体的操作和和选择操作的代码分开。
函数指针作为转换表
转换表就是一个函数指针数组。
#include<stdio.h>
#include<math.h>
// 转换表
// 转换表 step1:
//(1.1)声明 转台转移函数
double add(double, double);
double sub(double, double);
double mul(double, double);
double div(double, double);
double hypotenuse(double, double);
//(1.2)声明并初始化一个函数指针数组 pfunc:数组 数组元素:函数指针 返回值:double型数据
double(*pfunc[])(double, double) = { add, sub, mul, div, hypotenuse };//5个转移状态
//状态转移函数的实现
double add(double a, double b){ return a + b;}
double sub(double a, double b){ return a - b; }
double mul(double a, double b){ return a * b; }
double div(double a, double b){ return a / b; }
double hypotenuse(double a, double b){ return sqrt(pow(a, 2) + pow(b, 2)); }
void test()
{//转换表 step2:调用 函数指针数组int n = sizeof(pfunc) / sizeof(pfunc[0]);//转移表中 包含的元素个数(状态转移函数个数)for (int i = 0; i < n; ++i){printf("%.2lf\n",pfunc[i](3, 4));}
}
int main()
{test();return 0;
}
4、typedef
(1)使用typedef为现有类型创建别名,给变量定义一个易于记忆且意义明确的新名字。
-
类型过长,用typedef可以简化一下
typedef unsigned int UInt32
-
还可以定义数组类型
typedef int IntArray[10];
IntArray arr; //相当于int arr[10]
(2)使用typedef简化一些比较复杂的类型声明。
例如:
typedef int (*CompareCallBack)(int,int);
上述声明引入了PFUN类型作为函数指针的同义字,该函数有两个类型分别为int、int、char参数,以及一个类型为int的返回值。通常,当某个函数的参数是一个回调函数时,可能会用到typedef简化声明。 例如,承接上面的示例,我们再列举下列示例:
int callBackTest(int a,int b,CompareCallBack cmp);
callBackTest函数的参数有一个CompareCallBack类型的回调函数。在这个示例中,如果不用typedef,callBackTest函数声明如下:
int callBackTest(int a,int b,int (*cmp)(int,int));
从上面两条函数声明可以看出,不使用typedef的情况下,callBackTest函数的声明复杂得多,不利于代码的理解,并且增加的出错风险。
所以,在某些复杂的类型声明中,使用typedef进行声明的简化是很有必要的。
5、回调函数
首先要明确的一点是,函数也可以作为函数的参数来传递。
当做函数参数传入的函数,称之为 回调函数(至于为什么要叫“回调函数”,不能叫别的呢?其实这只是人为规定的一个名字。你也可以叫“maye专属函数”,但是到时候你又会问为什么要叫“maye专属函数”,它特么的总的有个名字吧!所以叫“回调函数”就是王八的屁股:规定!)。
实现一个与类型无关的查找函数
6、如何看懂复杂的指针
指针大家都学过了,简单的指针相信大家都不放在眼里,就不再赘述,但是复杂的你能理解吗?能理解指针就学的差不多了,至于如何运用只要你看懂指针就知道应该给它赋什么值,怎么用。
首先咱们一起来看看这个: int (*fun)(int *p)
首先需要分析这个是不是一个指针,如果是,是什么指针?如果不是,那是什么?
- 根据(*fun)可知,fun是一个指针
- 然后看fun的后面是一个函数参数列表,可以确定是一个指向函数的指针
- 指向的函数的返回值是什么类型呢,再回头看看最前面发现是一个int
- 最后我们可以根据这个函数指针写出对应的函数
结果如下:
int foo(int *p)
{return 0;
}
7、右左法则
上面我们分析了一个函数指针,那结果是如何得出来的呢?全靠经验吗,NO,其实是有方法的。
这个方法叫做右左法则:
右左法则不是C标准里面的内容,它是从C标准的声明规定中归纳出来的方法。C标准的声明规则,是用来解决如何创建声明的,而右左法则是用来解决如何辩识一个声明的。
右左法则使用:
- 首先从最里面的圆括号(应该是标识符)看起,然后往右看,再往左看;
- 每当遇到圆括号时,就应该调转阅读方向;
- 一旦解析完圆括号里面所有东西,就跳出圆括号;
- 重复这个过程知道整个声明解析完毕。
8、案例走起
(1)int (*p[5])(int*)
解析:
- 从标识符p开始,p先与[]结合形成一个数组,然后与*结合,表示是一个指针数组;
- 然后跳出这个圆括号,往后看,发现了一个函数的参数列表,说明数组里面装的是函数指针;
- 在跳出圆括号,往前看返回类型,可以确定函数指针的类型。
(2)int (*fun)(int *p,int (*pf)(int *))
解析:
- fun与*结合形成指针;
- 往后看是一个参数列表,说明是一个函数指针,只不过参数里面还有一个函数指针;
- 往前看可以确定函数指针的返回类型。
(3)int (*(*fun)[5])(int *p)
解析:
- fun与*结合,形成指针;
- 往后看发现了一个[5]说明是一个指向数组的指针;
- 再往前看,发现有一个*,说明数组里面存的是指针;
- 跳出圆括号往后看,发现了参数列表,说明数组里面存的是函数指针;
- 再往前看可以确定函数指针的返回类型。
(4)int (*(*fun)(int *p))[5]
解析:
- fun与*结合,形成指针;
- 往后看发现了参数列表,说明fun是一个函数指针;
- 往前看遇到了*说明,函数指针的返回类型是一个指针,是什么指针继续往后解析;
- 往后看发现了[5] 说明是一个数组指针,最前面一个int,说明fun这个函数指针的返回类型是一个数组的指针类型为int (*)[5]
(5)int(*(*fun())())()
解析:
- fun与()结合,说明fun是一个函数;
- 往前看发现了一个*,说明函数返回类型为指针,什么指针呢?
- 往后看发现了参数列表,fun函数返回的是一个函数指针,那这个函数指针的返回类型是什么呢?
- 往前看又发现了一个*,说明函数指针返回类型也是一个指针,那这个指针是什么指针呢?
- 往后看又发现了一个参数列表,说明是个函数指针,往前看这个函数指针返回的是int类型
9、总结
实际当中,需要声明一个复杂指针时,如果把整个声明写成上面所示的形式,对程序可读性是一大损害。应该用typedef来对声明逐层分解,增强可读性
指针变量有两种类型:指针变量的类型和指针所指向的对象的类型,指针变量的类型 只要把指针声明语句里的指针名字去掉,剩下的部分就是这个指针的类型。
-
int* ptr; //指针的类型是int
-
char* ptr; //指针的类型是char
-
int** ptr; //指针的类型是int**
-
int(*ptr)[3]; //指针的类型是int()[3]
-
int*(*ptr)[4]; //指针的类型是int*(*)[4]
指针变量指向的对象的类型
- 你只须把指针声明语句中的指针名字和名字左边的指针声明符*去掉,剩下的就是指针所指向的类型。
- int*ptr; //指针所指向的类型是int
- char*ptr; //指针所指向的的类型是char
- int**ptr; //指针所指向的的类型是int*
- int(*ptr)[3]; //指针所指向的的类型是int()[3]
- int*(*ptr)[4]; //指针所指向的的类型是int*()[4]
注意事项:
- 指针变量也是变量,也有存储空间,存的是别的变量的地址。
- 要注意指针的值,和指向的对象的值得区别
- 普通变量中的内存空间存放的是,数值或字符等。 ----直接存取
- 指针变量中的内存空间存放的是,另外一个普通变量的地址。----间接存取
- 连续定义多个指针变量时,容易犯错误,比如:int *p,p1;只有p是指针变量,p1是整型变量
- 避免使用为初始化的指针,很多运行错误都是由于这个原因导致的,而且这种错误又不能被编译器检查所以很难被发现,解决方法:初始化为NULL,报错就能很快找到原因
- 指针赋值时一定要保证类型匹配,由于指针类型确定指针所指向对象的类型,操作指针是才能知道按什么类型去操作
- 在用动态分配完内存之后一定要判断是否分配成功,分配成功后才能使用。
- 在使用完之后一定要释放,释放后必须把指针置为NULL
相关文章:
C语言指针常见问题汇总
我们在学C语言时,指针是我们最头疼的问题之一,针对C语言指针,博主根据自己的实际学到的知识以及开发经验,总结了以下使用C语言指针时常见问题。 1、指针做函数参数 学习函数的时候,讲了函数的参数都是值拷贝…...

Coremail邮件系统全新上线存档邮箱功能
邮箱积累邮件太多,搜索起来又慢又麻烦! 我的重要邮件忘记下载丢失了!14天自动删除太难了! 有没有可能重要邮件自动存档,解救一下“遗忘星”人? 在我们日常工作中,邮件是最经常使用的办公工具之一…...

Python绘图
1.二维绘图 a. 一维数据集 用 Numpy ndarray 作为数据传入 ply 1. import numpy as np import matplotlib as mpl import matplotlib.pyplot as pltnp.random.seed(1000) y np.random.standard_normal(10) print "y %s"% y x range(len(y)) print "x%s&q…...
【独家】华为OD机试 - 第K个最小码值的字母(C 语言解题)
最近更新的博客 华为od 2023 | 什么是华为od,od 薪资待遇,od机试题清单华为OD机试真题大全,用 Python 解华为机试题 | 机试宝典【华为OD机试】全流程解析经验分享,题型分享,防作弊指南)华为od机试,独家整理 已参加机试…...
整数反转(python)
题目链接: https://leetcode.cn/problems/reverse-integer/ 题目描述: 给你一个 32 位的有符号整数 x ,返回将 x 中的数字部分反转后的结果。 如果反转后整数超过 32 位的有符号整数的范围 [−231,231−1][−2^{31}, 2^{31} − 1][−231,231…...

【数据结构】二叉树与堆
文章目录1.树概念及结构1.1树的相关概念1.2树的结构2.二叉树概念及结构2.1相关概念2.2特殊的二叉树2.3二叉树的性质2.4二叉树的存储结构3.二叉树的顺序结构及实现3.1二叉树的顺序结构3.2堆的概念3.3堆的实现Heap.hHeap.c3.4堆的应用3.4.1 堆排序3.4.2 TOP-KOJ题最小K个数4.二叉…...

Git图解-常用命令操作-可视化
目录 一、前言 二、初始化仓库 2.1 设置用户名与邮箱 2.2 初始化仓库 三、添加文件 四、查看文件状态 五、查看提交日志 六、查看差异 七、版本回退 八、删除文件 九、分支管理 9.1 创建分支 9.2 切换分支 9.3 查看分支 9.4 合并分支 十、文件冲突 十一、转视…...
C语言-基础了解-20-typedef
typedef 一、typedef C 语言提供了 typedef 关键字,您可以使用它来为类型取一个新的名字。下面的实例为单字节数字定义了一个术语 BYTE: typedef unsigned char BYTE; 在这个类型定义之后,标识符 BYTE 可作为类型 unsigned char 的缩写&…...

Ubuntu系统升级16.04升级18.04
一、需求说明 作为Linux发行版中的后起之秀,Ubuntu 在短短几年时间里便迅速成长为从Linux初学者到实验室用计算机/服务器都适合使用的发行版,目前官网最新版本是22.04。Ubuntu16.04是2016年4月发行的版本,于2019年4月停止更新维护。很多软件支…...

CM6.3.2启用Kerberos(附问题解决)
基础准备支持JCE的jdk重新安装JCE的jdk(已正确配置跳过)删除/usr/java/下面的jdk,然后通过CM->管理->安全->安装Java无限制...重新安装后,配置Java(可选)主机->主机配置->搜java->Java主目录 配置路径主机->所有主机->设置->高级:Java配置Kerberos安…...
QML 动画(组合动画)
在QML中,可以把多个动画组合成一个单一的动画。 组合动画的类型: ParallelAnimation 动画同时进行(并行)SequentialAnimation 动画按照顺序执行(顺序执行)注意:将动画分组为“顺序动画”或“…...

【PHP代码注入】PHP代码注入漏洞
漏洞原理RCE为两种漏洞的缩写,分别为Remote Command/Code Execute,远程命令/代码执行PHP代码注入也叫PHP代码执行(Code Execute)(Web方面),是指应用程序过滤不严,用户可以通过HTTP请求将代码注入到应用中执行。代码注入(代码执行)…...
Python 常用语句同C/C++、Java的不同
文章目录前言1. 数字 int2. 字符 string3. 列表 List4. 元组 tuple5. 字典 dictionary6. 集合 set7. 值类型变量与引用类型变量8. if elif else9. >、<、>、<、、!10. while11. for前言 本篇为本人前段时间的一个简单汇总,这里可能并不齐全,…...

一把火烧掉了苹果摆脱中国制造的幻想,印度制造难担重任
这几年苹果不断推动印度制造,希望摆脱对中国制造的依赖,然而近期苹果在印度的一家代工厂发生大火却证明了苹果的这一计划遭受重大打击,印度制造根本就无法中国制造。一、印度制造屡屡发生幺蛾子苹果推动印度制造已有多年了,然而印…...
常用的 JavaScript 数组 API
以下是一些常用的 JavaScript 数组 API 的代码示例: 1、push() push(): 在数组末尾添加一个或多个元素,返回新的数组长度 const arr [1, 2, 3]; const newLength arr.push(4, 5); console.log(arr); // [1, 2, 3, 4, 5] console.log(newLength); //…...

海思3531a pjsip交叉编译
学习文档: PJSUA2 Documentation — PJSUA2 Documentation 1.0-alpha documentationhttps://www.pjsip.org/docs/book-latest/html/index.html ./configure --prefix/opensource/pjproject-2.12/build3531a \ --host/opt/hisi-linux/x86-arm/arm-hisi…...

《安富莱嵌入式周报》第305期:超级震撼数码管瀑布,使用OpenAI生成单片机游戏代码的可玩性,120通道逻辑分析仪,复古电子设计,各种运动轨迹函数源码实现
往期周报汇总地址:嵌入式周报 - uCOS & uCGUI & emWin & embOS & TouchGFX & ThreadX - 硬汉嵌入式论坛 - Powered by Discuz! 说明: 谢谢大家的关注,继续为大家盘点上周精彩内容。 视频版: https://www.bi…...

力扣-查找每个员工花费的总时间
大家好,我是空空star,本篇带大家了解一道简单的力扣sql练习题。 文章目录前言一、题目:1741. 查找每个员工花费的总时间二、解题1.正确示范①提交SQL运行结果2.正确示范②提交SQL运行结果3.正确示范③提交SQL运行结果4.正确示范④提交SQL运行…...

企业级信息系统开发学习笔记1.8 基于Java配置方式使用Spring MVC
文章目录零、本节学习目标一、基于Java配置与注解的方式使用Spring MVC1、创建Maven项目 - SpringMVCDemo20202、在pom.xml文件里添加相关依赖3、创建日志属性文件 - log4j.properties4、创建首页文件 - index.jsp5、创建Spring MVC配置类 - SpringMvcConfig6、创建Web应用初始…...
【C语言复习】C语言中的文件操作
C语言中的文件操作写在前面文件操作什么是文件文件的分类文件名文件的操作文件指针文件的打开和关闭文件的顺序读写文件的随机读写fseekftellrewindfeof写在前面 文件操作在C语言部分只是属于了解内容,但是因为它可能会应用在项目中,所以我把它单独写成…...
Frozen-Flask :将 Flask 应用“冻结”为静态文件
Frozen-Flask 是一个用于将 Flask 应用“冻结”为静态文件的 Python 扩展。它的核心用途是:将一个 Flask Web 应用生成成纯静态 HTML 文件,从而可以部署到静态网站托管服务上,如 GitHub Pages、Netlify 或任何支持静态文件的网站服务器。 &am…...

2025盘古石杯决赛【手机取证】
前言 第三届盘古石杯国际电子数据取证大赛决赛 最后一题没有解出来,实在找不到,希望有大佬教一下我。 还有就会议时间,我感觉不是图片时间,因为在电脑看到是其他时间用老会议系统开的会。 手机取证 1、分析鸿蒙手机检材&#x…...
鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个生活电费的缴纳和查询小程序
一、项目初始化与配置 1. 创建项目 ohpm init harmony/utility-payment-app 2. 配置权限 // module.json5 {"requestPermissions": [{"name": "ohos.permission.INTERNET"},{"name": "ohos.permission.GET_NETWORK_INFO"…...
Java入门学习详细版(一)
大家好,Java 学习是一个系统学习的过程,核心原则就是“理论 实践 坚持”,并且需循序渐进,不可过于着急,本篇文章推出的这份详细入门学习资料将带大家从零基础开始,逐步掌握 Java 的核心概念和编程技能。 …...

分布式增量爬虫实现方案
之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面,避免重复抓取,以节省资源和时间。 在分布式环境下,增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路:将增量判…...

有限自动机到正规文法转换器v1.0
1 项目简介 这是一个功能强大的有限自动机(Finite Automaton, FA)到正规文法(Regular Grammar)转换器,它配备了一个直观且完整的图形用户界面,使用户能够轻松地进行操作和观察。该程序基于编译原理中的经典…...
怎么让Comfyui导出的图像不包含工作流信息,
为了数据安全,让Comfyui导出的图像不包含工作流信息,导出的图像就不会拖到comfyui中加载出来工作流。 ComfyUI的目录下node.py 直接移除 pnginfo(推荐) 在 save_images 方法中,删除或注释掉所有与 metadata …...
第7篇:中间件全链路监控与 SQL 性能分析实践
7.1 章节导读 在构建数据库中间件的过程中,可观测性 和 性能分析 是保障系统稳定性与可维护性的核心能力。 特别是在复杂分布式场景中,必须做到: 🔍 追踪每一条 SQL 的生命周期(从入口到数据库执行)&#…...

Rust 开发环境搭建
环境搭建 1、开发工具RustRover 或者vs code 2、Cygwin64 安装 https://cygwin.com/install.html 在工具终端执行: rustup toolchain install stable-x86_64-pc-windows-gnu rustup default stable-x86_64-pc-windows-gnu 2、Hello World fn main() { println…...

渗透实战PortSwigger靶场:lab13存储型DOM XSS详解
进来是需要留言的,先用做简单的 html 标签测试 发现面的</h1>不见了 数据包中找到了一个loadCommentsWithVulnerableEscapeHtml.js 他是把用户输入的<>进行 html 编码,输入的<>当成字符串处理回显到页面中,看来只是把用户输…...