第四站:指针的进阶-(二级指针,函数指针)
目录
二级指针
二级指针的用途
多级指针的定义和使用
指针和数组之间的关系
存储指针的数组(指针数组:保存地址值)
指向数组的指针(数组指针)
传参的形式(指针)
数组传参时会退化为指针
void类型的指针
函数指针
定义:
调用:两种方式:(*指针名)(参数地址) 或者 指针名(参数地址)
应用:
二级指针
二级指针也是一个普通的指针变量,只是它里面保存的值是另外一个一级指针的地址
#include <iostream>using namespace std;
int main(void) {int massage = 888;int* guizi1 = &massage;int** guizi2 = &guizi1;cout << "massage的地址" << &massage << endl;cout << "guizi1内的地址" << guizi1 << endl;cout << "guizi2内的地址" << *guizi2 << endl;cout << "guizi1内的值" << *guizi1 << endl;cout << "guizi2内的值" << **guizi2 << endl;return 0;
}
二级指针和一级指针的类型都是相同的,
二级指针所指向的对象就是一级指针的值,一级指针的值又是指向的变量的值,所以不管有多少级指针,他们只要保存的是上一级指针的地址,那么内部所保存的地址值都是一样的,
这里不要和指针变量本身作为一个变量的指针混淆了,所定义的二级指针和一级指针他们的地址值肯定是不同的但是他们内部所保存的内容(对象的地址)是相同的
二级指针的用途
普通指针可以将变量通过参数“带入”函数内部然后进行修改,但没办法将函数的内部变量“带出”函数
二级指针不但可以将变量通过参数带入函数内部,也可以将函数内部变量 “带出”到函数外部。
使用的一级指针报警:读取位置 0x0000000000000000 时发生访问冲突。这个全部为0的地址就是给一级指针初始化时设的NULL值(空指针,就是值为 0 的指针,任何程序数据都不会存储在地址为 0 的内存块中,它是被操作系统预留的内存块)
#include <iostream>using namespace std;void swap(int* a, int* b,int *c) {int tmp = *a;*a = *b;*b = tmp;int d = *a + *b;c = &d;}void boy(int** meipo) {static int boy = 22;*meipo = &boy;
}
int main(void) {int a = 8;int b = 3;int* c = NULL;swap(&a, &b, c);cout << "一级指针传入之后c的值:"<<endl;int* type = NULL;boy(&type);cout << "通过二级指针带回来的值:" << *type << endl;return 0;
}
通过程序调用可以看出给二级指针传入地址值后,二级指针可以通过一级指针把函数内部的变量地址值给带回来
多级指针的定义和使用
#include <iostream>using namespace std;
int main(void) {int massage = 888;int* guizi1 = &massage;int** guizi2 = &guizi1;int*** guizi3 = &guizi2;int**** guizi4 = &guizi3;int***** guizi5 = &guizi4;cout << "massage的地址" << &massage << endl;cout << "guizi1内的地址" << guizi1 << endl;cout << "guizi2内的地址" << *guizi2 << endl;cout << "guizi3内的地址" << **guizi3 << endl;cout << "guizi4内的地址" << ***guizi4 << endl;cout << "guizi5内的地址" << ****guizi5 << endl;cout << "guizi1内的值" << *guizi1 << endl;cout << "guizi2内的值" << **guizi2 << endl;cout << "guizi3内的值" << ***guizi3 << endl;cout << "guizi4内的值" << ****guizi4 << endl;cout << "guizi5内的值" << *****guizi5 << endl;return 0;
}
指针和数组之间的关系
#include <iostream>using namespace std;void print_1(int days[], int len) {for (int i = 0; i < len; i++) {//用数组的方式输出cout << "用数组的方式输出:" << days[i] << " ";//用指针的形式输出cout << "用指针的方式输出:" << *(days + i);cout << endl;}
}
void print_2(int* day, int len) {for (int i = 0; i < len; i++) {//用数组的方式输出cout << "用数组的方式输出:" << day[i] << " ";//用指针的形式输出cout << "用指针的方式输出:" << *(day + i);cout << endl;}
}
int main(void) {int days[] = { 31,28,31,30,31,30,31,31,30,31,30,31 };int* day = NULL;int len = sizeof(days) / sizeof(int);day = days;//for (int i = 0; i < len; i++){// //用数组的方式输出// cout << "用数组的方式输出:" << days[i] <<" ";// //用指针的形式输出// cout << "用指针的方式输出:" << *(day + i);// cout << endl;//}//当传入参数是数组时cout << "当传入参数是数组时:" << endl;print_1(days, len);cout << "当传入参数是指针时:" << endl;print_2(day, len);return 0;
}
存储指针的数组(指针数组:保存地址值)
定义: 类型 *指针数组名[元素个数] ;如:int *p[2]
定义一个有n个元素个数指针数组,每个元素都是一个指针变量
指针数组是先有地址值,然后再通过地址值,指针数组本质可以理解为不是一个指针而是一个只保存地址的数组,在操作指针数组时,这个指针数组是先保存有这个对象的地址(并不是指向这个对象),然后可以通过地址访问这个对象的值
#include <iostream>using namespace std;int main(void) {int days[] = { 31,28,31,30,31,30,31,31,30,31,30,31 };int len = sizeof(days) / sizeof(int);int* p[1];//代表这个指针数组内,有一个指针变量p[0] = days;for (int i = 0; i < len; i++){if (*p[0] > days[i]) {p[0] = &days[i];}}cout << *p[0] << endl;return 0;
}
指向数组的指针(数组指针)
1. 指向数组的指针
int (*p)[3]; //定义一个指向三个成员的数组的指针,数组指针的个数,和数组的列数是对应的
访问元素的两种方式:
数组法: (*p)[j]
指针法: *((*p)+j)
#include <iostream>using namespace std;int main(void) {int days[4][3] = {31,28,31,30,31,30,31,31,30,31,30,31};//几维的数据就将元素设为几个元素的指针int (*p)[3];//定义一个指针数组,指向4个元素的的数组的指针int* little = NULL;//使用数组型的指针型数组访问p = &days[0];for (int i = 0; i < 4; i++){for (int j = 0; j < 3; j++){cout << (*p)[j] << " ";}p++;}cout << endl;//使用指针型的指针数组访问p = &days[0];little = &(*p)[0];for (int i = 0; i < 4; i++) {for (int j = 0; j < 3; j++) {if (*little > *((*p) + j)) {little = (*p) + j;}}p++;}cout << *little << endl;return 0;
}
传参的形式(指针)
数组传参时会退化为指针
1)退化的意义:C 语言只会以值拷贝的方式传递参数,参数传递时,如果拷贝整个数组,效率会大大降低,并且参数将位于栈上,太大的数组拷贝将会导致栈溢出。
2)因此,C 语言将数组的传参进行了退化。将整个数组拷贝一份传入函数时,将数组名看做常量指针,传数组首元素的地址。(这样参数在进行赋值时,就会和这个数组共用同一个内存空间)
#include <iostream>using namespace std;
//方法1
void method_1(int days[12]) {for (int i = 0; i < 12; i++){cout << days[i] << " ";}
}
//方法2
void method_2(int days[], int len) {for (int i = 0; i < len; i++) {cout << days[i] << " ";}
}
//方法3
void method_3(int* array, int len) {for (int i = 0; i < len; i++) {cout << *(array+i) << " ";}
}
//方法4
void method_4(int* arr[]) {for (int i = 0; i < 12; i++) {cout << *arr[i] << " ";}
}
//方法5
void method_5(int* arr[], int len) {for (int i = 0; i < len; i++) {cout << *arr[i] << " ";}
}
//方法6
void method_6(int** arr, int len) {for (int i = 0; i < len; i++) {cout << *arr[i] << " ";}
}
//方法7
void method_7(int(*k)[12]) {for (int i = 0; i < 12; i++) {cout << *(*k+i) << " ";}
}
int main(void) {//数组作为参数传递,输出数组的6种方式int days[12] = {31,28,31,30,31,30,31,31,30,31,30,31};int len = sizeof(days) / sizeof(int);//方法一;cout << "方法一:将数组作为形参参接收数组:" << endl;method_1(days);cout << endl;cout << "方法二:将数组作为形参参接收数组,并指定数组的元素个数:" << endl;method_2(days, len);cout << endl;cout << "方法三:使用指针作为形参接受数组,并指定数组的元素个数:" << endl;method_3(days, len);cout << endl;cout << "方法四:使用指针数组作为形参:" << endl;//先为指针数组内的元素赋值int* day[12] = {0};//初始化,指针数组时先作为一个数组for (int i = 0; i < len; i++){day[i] = &days[i];}method_4(day);cout << endl;cout << "方法五:使用指针数组作为形参,并指定数组的元素个数:" << endl;method_5(day, len);cout << endl;cout << "方法六:使用二级指针作为形参,并指定数组的元素个数:" << endl;method_6(day, len);cout << endl;int(*k)[12];k = &days;cout << "方法七:使用数组指针作为形参接受数组:" << endl;method_7(k);return 0;
}
void类型的指针
void => 空类型
void* => 空类型指针,只存储地址的值,丢失类型,无法访问,要访问其值,我们必须对这个指 针做出正确的类型转换(其他类型赋值给void指针,会自动转,void的指针赋值给其他类型则需要强制转换),然后再间接引用指针。
void*类型的指针是无法进行算数运算的,同样无法访问(E0852 表达式必须是指向完整对象类型的指针)
#include <iostream>using namespace std;int main(void) {int a = 10;int* p = &a;void* p1 = &a;void* p2 = p; //其他类型的指针赋值给void类型的指针自动转换p++;//有具体类型的指针可以算术运算//p1++;//错误int* p3 = (int*)p1;//void类型的指针赋值给其他类型的指针需要进行强制转换//cout << *p1 << endl;//E0852:表达式必须是指向完整对象类型的指针cout << p1 << endl;//可以输出他的地址值cout << p2 << endl;cout << *p3 << endl;return 0;
}
函数指针
函数指针简单理解就是把整个函数当做一个指针
定义:
把函数声明移过来,把函数名改成 (* 函数指针名)
例如:
定义一个函数为:int man(const void *a,const void *b){}
函数的指针定义为:int (*p)(const void *a ,const void *b);
最后将函数的地址赋值给指针:p = &man;
调用:
两种方式:(*指针名)(参数地址) 或者 指针名(参数地址)
例如上述:(*p)(&a,&b); 或者 p(&a,&b);
#include <iostream>using namespace std;
int compare( void* a, void* b) {int* x = (int*)a;int* y = (int*)b;return *x - *y;}int main(void) {int arr[] = { 2, 10, 30, 1, 11, 8, 7,25 };int len = sizeof(arr) / sizeof(int);int (*p)( void* a, void* b);p = &compare;cout << arr[0] << " " << arr[1] << endl;cout << p(&arr[0], &arr[1]) << endl;return 0;
}
应用:
使用VS自带的快速排序函数,来调用自定函数进行排序
#include <iostream>using namespace std;
int compare(const void* a,const void* b) {int *x = (int*)a;int *y = (int*)b;return *x - *y;//小于0,升序排序,
}int main(void) {int arr[] = { 2, 10, 30, 1, 11, 8, 7,25 };int len = sizeof(arr) / sizeof(int);cout << "排序前:" << endl;for (int i = 0; i < len; i++){cout << arr[i] << " ";}int (*p)(const void* a,const void* b);p = &compare;//这里qsort函数的第四个参数需要的函数参数类型为const类型的指针qsort(arr, len, sizeof(int), p);cout << endl;//这个作用是换行cout <<"排序后:" << endl;for (int i = 0; i < len; i++) {cout << arr[i] << " ";}return 0;
}
相关文章:

第四站:指针的进阶-(二级指针,函数指针)
目录 二级指针 二级指针的用途 多级指针的定义和使用 指针和数组之间的关系 存储指针的数组(指针数组:保存地址值) 指向数组的指针(数组指针) 传参的形式(指针) 数组传参时会退化为指针 void类型的指针 函数指针 定义: 调用:两种方式:(*指针名)(参数地址) 或者 指针…...
浏览器渲染原理(面试重点)
一、浏览器是如何渲染页面的 常见的简洁答案: 浏览器内核拿到内容后,渲染流程大致如下:解析HTML,构建Dom树;解析CSS,构建Render树;(将CSS代码解析成树形的数据结构,与D…...
C //练习 5-3 用指针方式实现第2章中的函数strcat。函数strcat(s, t)将t指向的字符串复制到s指向的字符串的尾部。
C程序设计语言 (第二版) 练习 5-3 练习 5-3 用指针方式实现第2章中的函数strcat。函数strcat(s, t)将t指向的字符串复制到s指向的字符串的尾部。 注意:代码在win32控制台运行,在不同的IDE环境下,有部分可能需要变更。…...
深度剖析Redis:从基础到高级应用
目录 引言 1、 Redis基础 1.1 Redis数据结构 1.1.1 字符串(String) 1.1.2 列表(List) 1.1.3 集合(Set) 1.1.4 散列(Hash) 1.1.5 有序集合(Sorted Set)…...

视频监控录像服务器(中心录像服务器)功能详细介绍
目 录 一、概述 (一)定义 (二)视频监控中心录像服务器 二、存储策略服务 (一)存储策略配置 1、 录入页面 2、 选择需要进行录像的视频 3、批量选择多个通道号 4、其他关键参数…...

SouthernBiotech抗荧光淬灭封片剂
荧光淬灭又称荧光熄灭或萃灭,是指导致特定物质的荧光强度和寿命减少的所有现象。引起荧光淬灭的物质称为荧光淬灭剂。SouthernBiotech专门开发的Fluoromount-G系列荧光封片剂是以甘油为基础,加入抗荧光淬灭剂,可明显降低荧光淬灭现象…...

[Excel]如何找到非固定空白格數列的條件數據? 以月份報價表單為例
在群組中看到上述問題,研判應是一份隨月份變動的產品報價表單,空白欄可能表示該月份價格與上個月份一致。這個問題是需要取得最近一次單價和倒數第二次單價,常用且實務的excel案例值得紀錄。 最近一次單價: INDEX($B2:$G2,1,LARGE(IF(ISBLAN…...

TypeScript进阶(二)深入理解装饰器
✨ 专栏介绍 TypeScript是一种由微软开发的开源编程语言,它是JavaScript的超集,意味着任何有效的JavaScript代码都是有效的TypeScript代码。TypeScript通过添加静态类型和其他特性来增强JavaScript,使其更适合大型项目和团队开发。 在TypeS…...

书生·浦语第三次作业
我最近在参加书生浦语大模型实战营,这是第三次作业打卡! 如果你也想两周玩转大模型微调,部署与测评全链路。报名链接:invite 书生浦语大模型实战营报名 邀请码可以填026014 一、基础作业:复现课程知识库助手搭建过程…...
GPT实战系列-LangChain + ChatGLM3构建天气查询助手
GPT实战系列-LangChain ChatGLM3构建天气查询助手 用ChatGLM的工具可以实现很多查询接口和执行命令,而LangChain是很热的大模型应用框架。如何联合它们实现大模型查询助手功能?例如调用工具实现网络天气查询助手功能。 LLM大模型相关文章: …...
LeetCode 2696.删除子串后的字符串最小长度:栈
【LetMeFly】2696.删除子串后的字符串最小长度:栈 力扣题目链接:https://leetcode.cn/problems/minimum-string-length-after-removing-substrings/ 给你一个仅由 大写 英文字符组成的字符串 s 。 你可以对此字符串执行一些操作,在每一步操…...

Xcode15 升级问题记录
这里写自定义目录标题 新版本Xcode15升级问题1:rsync error: some files could not be transferred (code 23) at ...参考 新版本Xcode15升级 下载地址:https://developer.apple.com/download/all/ 我目前使用的版本是Xcode15.2 我新创建了一个项目&…...
List、Set、Map有什么区别?
List、Set和Map是Java中的三种基本数据结构,它们在元素重复性、有序性和用途方面存在显著的区别。 元素重复性: List允许有重复的元素。任何数量的重复元素都可以在不影响现有重复元素的值及其索引的情况下插入到List集合中。 Set集合不允许元素重复。…...

centOS系统yum安装和卸载mongodb
0.1 什么是mongodb? 0.2 Mongodb是一个基于分布式文件存储的数据库。由C语言编写。旨在为WEB应用提供可扩展的高性能数据存储解决方案。 0.3 Mongodb是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据…...

2023年12月 C/C++(一级)真题解析#中国电子学会#全国青少年软件编程等级考试
C/C++编程(1~8级)全部真题・点这里 第1题:数的输入和输出 输入一个整数和双精度浮点数,先将浮点数保留2位小数输出,然后输出整数。 时间限制:1000 内存限制:65536 输入 一行两个数,分别为整数N(不超过整型范围),双精度浮点数F,以一个空格分开。 输出 一行两个数,分…...

Python爬虫---Scrapy项目的创建及运行
Scrapy是一个为了爬取网站数据,提取结构性数据而编写的应用框架。 可以应用在包括数据挖 掘,信息处理或存储历史数据等一系列的程序中。 1. 安装scrapy: pip install scrapy 注意:需要安装在python解释器相同的位置,例如…...
PyTorch: torch.nn 子模块及其在循环神经网络中的应用
目录 torch.nn子模块详解 nn.utils.rnn.PackedSequence 参数说明 注意事项 示例代码 nn.utils.rnn.pack_padded_sequence 参数说明 返回值 注意事项 示例代码 nn.utils.rnn.pad_packed_sequence 参数说明 返回值 注意事项 示例代码 nn.utils.rnn.pad_sequence …...

【QT】自定义代理类
目录 1 我们为什么要使用自定义代理类? 2 自定义代理类的基本设计要求 3 自定义代理的功能 4 基于QSpinBox的自定义代理类 5 自定义代理类的使用 1 我们为什么要使用自定义代理类? 传统的模型-视图框架可以让我们实现逻辑展示相分离,我们…...
线程休眠、线程让步、线程优先级相关内容学习笔记
1、线程休眠 (1)sleep() 如果需要让当前正在执行的线程暂停一段时间,并进入阻塞状态(Timed_Waiting),则可以通过调用Thread类的静态sleep()方法来实现。 static void sleep(long millis):让当前正在执行的线…...
paddle指定运行gpu
在PaddlePaddle中指定使用GPU进行运行非常简单。首先,确保你的机器上已经安装了CUDA和cuDNN,并且正确配置了GPU环境。然后,按照以下步骤进行操作: 导入PaddlePaddle库: import paddle设置使用的设备为GPU:…...

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析
1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具,该工具基于TUN接口实现其功能,利用反向TCP/TLS连接建立一条隐蔽的通信信道,支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式,适应复杂网…...

HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...
数据库分批入库
今天在工作中,遇到一个问题,就是分批查询的时候,由于批次过大导致出现了一些问题,一下是问题描述和解决方案: 示例: // 假设已有数据列表 dataList 和 PreparedStatement pstmt int batchSize 1000; // …...

Mac下Android Studio扫描根目录卡死问题记录
环境信息 操作系统: macOS 15.5 (Apple M2芯片)Android Studio版本: Meerkat Feature Drop | 2024.3.2 Patch 1 (Build #AI-243.26053.27.2432.13536105, 2025年5月22日构建) 问题现象 在项目开发过程中,提示一个依赖外部头文件的cpp源文件需要同步,点…...

SAP学习笔记 - 开发26 - 前端Fiori开发 OData V2 和 V4 的差异 (Deepseek整理)
上一章用到了V2 的概念,其实 Fiori当中还有 V4,咱们这一章来总结一下 V2 和 V4。 SAP学习笔记 - 开发25 - 前端Fiori开发 Remote OData Service(使用远端Odata服务),代理中间件(ui5-middleware-simpleproxy)-CSDN博客…...

面向无人机海岸带生态系统监测的语义分割基准数据集
描述:海岸带生态系统的监测是维护生态平衡和可持续发展的重要任务。语义分割技术在遥感影像中的应用为海岸带生态系统的精准监测提供了有效手段。然而,目前该领域仍面临一个挑战,即缺乏公开的专门面向海岸带生态系统的语义分割基准数据集。受…...
多模态图像修复系统:基于深度学习的图片修复实现
多模态图像修复系统:基于深度学习的图片修复实现 1. 系统概述 本系统使用多模态大模型(Stable Diffusion Inpainting)实现图像修复功能,结合文本描述和图片输入,对指定区域进行内容修复。系统包含完整的数据处理、模型训练、推理部署流程。 import torch import numpy …...

逻辑回归暴力训练预测金融欺诈
简述 「使用逻辑回归暴力预测金融欺诈,并不断增加特征维度持续测试」的做法,体现了一种逐步建模与迭代验证的实验思路,在金融欺诈检测中非常有价值,本文作为一篇回顾性记录了早年间公司给某行做反欺诈预测用到的技术和思路。百度…...
HTML前端开发:JavaScript 获取元素方法详解
作为前端开发者,高效获取 DOM 元素是必备技能。以下是 JS 中核心的获取元素方法,分为两大系列: 一、getElementBy... 系列 传统方法,直接通过 DOM 接口访问,返回动态集合(元素变化会实时更新)。…...

Matlab实现任意伪彩色图像可视化显示
Matlab实现任意伪彩色图像可视化显示 1、灰度原始图像2、RGB彩色原始图像 在科研研究中,如何展示好看的实验结果图像非常重要!!! 1、灰度原始图像 灰度图像每个像素点只有一个数值,代表该点的亮度(或…...