进阶指针(一)
✨博客主页:小钱编程成长记
🎈博客专栏:进阶C语言
进阶指针(一)
- 0.回顾初阶指针
- 1.字符指针
- 1.1 相关面试题
- 2.数组指针
- 3.指针数组
- 3.1 数组指针的定义
- 3.2 &数组名VS数组名
- 3.3 数组指针的使用
- 4.数组传参和指针传参
- 4.1 一维数组传参
- 4.2 二维数组传参
- 4.3 一级指针传参
- 4.4 二级指针传参
- 5.函数指针
- 5.1 两段有趣的代码
- 总结
0.回顾初阶指针
- 指针就是个变量,用来存放地址,地址唯一标识一块内存空间。(内存单元数有编号的,编号=地址=指针)
- 指针的大小是固定的4/8个字节(32位平台/64位平台)。
- 指针是有类型的,指针的类型决定了指针的±整数的步长,指针解引用操作的时候的权限。
- 指针的运算。
1.字符指针
在指针的类型中我们知道有一种指针类型为字符指针 char* ;
它有两种使用方式:
- 指向字符
- 指向字符串(实际上指向的是首字符,但因为字符串中的字符都是连续的,所以也可以说是指向字符串)
指针指向字符:
int main()
{char ch = 'w';char *pc = &ch;*pc = 'w';return 0;
}
指针指向字符串:
int main()
{const char* pstr = "abcdef";//这里是把一个字符串放到pstr指针变量里了吗?printf("%s\n", pstr);return 0;
}
-
const char* pstr = “abcdef”;
这里不是把一个字符串放到pstr指针变量里,而是将字符串的首字符的地址放到了pstr里。因为当字符串作为一个表达式时,结果是首字符的地址。 -
const *pstr = “abcdef” 和 char arr[] = “abcdef” 在内存中存储的都是abcdef\0 ;
-
因为"abcdef” 和 arr 表示的都是字符串的首字符地址,所以我们可以将常量字符串想象成数组名,“abcdef” == arr。
如下所示:
有些朋友可能会发现,为什么指针指向常量字符串时前面要加上const?比如:const char* pstr = “abcdef”;
原因是:常量字符串在内存中不能被修改,若修改会出现写入错误(这个错误很难被及时发现)。如下所示:
用const修饰指针变量,使其变成常变量,不能被修改,即使不小心修改了,也能在编译期间及时发现错误。如下所示:
1.1 相关面试题
//在《剑指offer》一书中有这样一道题
#include <stdio.h>
int main()
{char str1[] = "hello";char str2[] = "hello";const char* str3 = "hello";const char* str4 = "hello";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;
}
结果为:
- 把常量字符串放到字符数组中,字符数组中存放的是字符。
每个数组创建时在内存中开辟的空间并不同,所以用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。(几个 数组的内容可能会相同,但每个数组在内存中的地址一定不同)
所以str1和str2代表的首字符地址不同。- C/C++会把常量字符串存储到一个单独的内存区域。因为常量字符串不能被修改,没必要保存多份,所以在内存中只存储一份。
当几个指针指向同一个字符串时,实际上它们指向的也是同一个地址。
2.数组指针
在初阶指针中我们也学了指针数组。指针数组是一个存放指针的数组,存放在数组中的元素都是指针类型的。
我们来复习一下下面的指针数组是什么意思?
int* arr1[10]; //整形指针的数组
char *arr2[4]; //一级字符指针的数组
char **arr3[5];//二级字符指针的数组
指针数组的使用场景举例:
//模拟二维数组
#include <stdio.h>
int main()
{int arr1[] = { 1,2,3,4,5 };int arr2[] = { 2,3,4,5,6 };int arr3[] = { 3,4,5,6,7 };int* arr[] = { arr1, arr2, arr3 };int i = 0;for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 5; j++){printf("%d ", arr[i][j]);}printf("\n");}return 0;
}//一次性打印多个字符串
#include <stdio.h>
int main()
{char* arr[3] = { "hello", "hello", "C++" };int i = 0;for (i = 0; i < 3; i++){printf("%s ", arr[i]);}return 0;
}
3.指针数组
3.1 数组指针的定义
数组指针是指针?还是数组?
答案是:指针。
我们已经知道:
整型指针: int * pint; 能够指向整型数据的指针。
浮点型指针: float * pf; 能够指向浮点型数据的指针。
那数组指针应该是:能够指向数组的指针。
下面代码哪个是数组指针?
int *p1[10];//p1是指针数组名
int (*p2)[10];//p2是数组指针
解释:
int (*p)[10];
//解释:p先和*结合,说明p是一个指针变量,然后指向的是一个大小为10个整型的数组。所以p是一个指针,指向一个数组,叫数组指针。
//注意:必须明确指针指向的数组有几个元素,不写时,默认为0,与真实数组的元素个数不同,会出错。
//指向的数组的元素个数不同,数组指针的类型也不同。
3.2 &数组名VS数组名
arr和&arr分别是什么?
我们知道arr是数组名,数组名表示的是数组的首元素地址。
那&arr数组名是什么呢?
我们先来看一段代码和运行结果:
数组名和&数组名打印的地址是一样的。
难道两个是一样的吗?
我们再来看一段代码和运行结果:
根据上面的代码我们发现,其实&arr和arr,虽然值是一样的,但是意义是不一样的。
实际上: &arr 表示的是数组的地址,而不是数组首元素的地址。
本例中 &arr 的类型是: int(*)[10] ,是一种数组指针类型, 数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于 &arr 的差值是40
3.3 数组指针的使用
数组指针指向的是数组,那么数组指针中存放的应该是数组的地址。
#include <stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };int(*p)[10] = &arr;//把数组arr的地址赋值给数组指针变量preturn 0;
}
小提示:
printf("%d\n", (*p)[i]);//(*p) == *(&arr) == arr
printf("%d\n", p[i]);//p[i] == *(p+i) 因为p = &arr,所以p+i等于跳过了i个数组
数组指针主要应用于二维数组的传参:
#include <stdio.h>void print_arr1(int(*arr)[5], int row, int col)//arr是数组指针
{int i = 0;for (i = 0; i < row; i++){int j = 0;for (j = 0; j < col; j++){printf("%d ", arr[i][j]);//arr[i][j] == (*(arr+i))[j], //arr+i相当于二维数组的第i行的一维数组的地址,//*(arr+i)相当于二维数组的第i行的一维数组的首元素地址。}printf("\n");}
}int main()
{int arr[3][5] = { 1,2,3,4,5,6,7,8,9,10 };print_arr1(arr, 3, 5);//数组名arr,表示首元素的地址//但是二维数组的首元素是二维数组的第一行//所以这里传递的arr,其实相当于第一行的地址,是一维数组的地址//可以数组指针来接收return 0;
}
4.数组传参和指针传参
4.1 一维数组传参
#include <stdio.h>
void test(int arr[])//形参是数组的形式,但并不会真正创建一个数组,所以大小没有意义,可以随便写,也可以不写。数组形式的本质还是指针。
{}
void test(int arr[10])//形参是数组
{}
void test(int* arr)//形参是指针
{}void test2(int* arr[20])//形参是指针数组
{}
void test2(int** arr)//形参是指针(元素)的指针,二级指针。
{}
int main()
{int arr[10] = { 0 };int* arr2[20] = { 0 };test(arr);test2(arr2);
}
4.2 二维数组传参
void test(int arr[3][5])
{}
void test(int arr[][5])//行可以省略,但列不能省略,因为若不确定列,数据连续存储后,就无法正确拆开连续存储的数据排成几行输出。
{}void test(int(*arr)[5])//因为二维数组的首元素是一维数组,所以形参用指针时要用数组指针。
{}int main()
{int arr[3][5] = { 0 };test(arr);
}
4.3 一级指针传参
#include <stdio.h>
void print(int* p, int sz)
{int i = 0;for (i = 0; i < sz; i++){printf("%d\n", *(p + i));}
}
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9 };int* p = arr;int sz = sizeof(arr) / sizeof(arr[0]);//一级指针p,传给函数,形参写成一级指针就行了。print(p, sz);return 0;
}
思考:
当一个函数的参数部分为一级指针的时候,函数能接收什么参数?
例如:
void test1(int *p)//test1函数能接收什么参数?
{}int main()
{int a = 10;test1(&a);//传整型变量的地址int* pa = &a;test1(pa);//传整型指针int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };test1(arr);//传整形一维数组的数组名return 0;
}
4.4 二级指针传参
#include <stdio.h>
void test(int** ptr)
{printf("num = %d\n", **ptr);
}
int main()
{int n = 10;int* p = &n;int** pp = &p;test(pp);test(&p);return 0;
}
思考:
当函数的参数为二级指针的时候,可以接收什么参数?
例如:
void test(char** p)
{}int main()
{char c = 'b';char* pc = &c;char** ppc = &pc;char* arr[10];test(&pc);//指针的地址test(ppc);//二级指针test(arr);//指针数组的数组名:首元素(指针)的地址return 0;
}
5.函数指针
我们先来看一段代码:
#include <stdio.h>int Add(int x, int y)
{return x + y;
}int main()
{printf("%p\n", &Add);printf("%p\n", Add);return 0;
}
由此可见:
&函数名和函数名都是函数的地址
int (*pf1)(int, int) = Add;//pf1就是函数指针变量
pf1先和*结合,说明pf1是指针,指向的是一个函数,指向的函数有两个参数,参数类型都是int,返回类型为int
举例:
#include <stdio.h>int Add(int x, int y)
{return x + y;
}int main()
{int (*pf1)(int, int) = &Add;int ret1 = (*pf1)(2, 3);//函数名是地址,地址也是函数名,所以写不写*都行,*几乎是个摆设,写几个都行。int ret11 = (pf1)(2, 3);//int ret3 = &Add(2, 3);错误,因为 & 取的内容,必须是 = 左边出现过的。int (*pf2)(int, int) = Add;int ret2 = (*pf2)(2, 3);int ret22 = (pf2)(2, 3);int ret33 = Add(2, 3);printf("%d\n", ret1);printf("%d\n", ret11);printf("%d\n", ret2);printf("%d\n", ret22);printf("%d\n", ret33);return 0;
}
int ret3 = &Add(2, 3);
错误,因为 & 取的内容,必须是 = 左边出现过的。
5.1 两段有趣的代码
《C陷阱和缺陷》一书中提及这两个代码:
代码1:
(*(void (*)())0)();
void (*)()是一个函数指针类型,(void ( * )())0 是把0强转成这种函数指针类型的数据。这个代码是用来调用0地址处的函数。这个函数没有参数,返回类型是void。( *函数地址0的操作可写可不写,因为函数地址也就相当于函数名)
疑问:代码2:
void (*signal(int, void(*)(int)))(int);
- 这个代码是一次函数声明,声明的是signal函数,signal函数的参数有2个;第一个是int类型,第二个是函数指针类型,该类型是void ( * )(int);该函数指针指向的函数,参数是int类型的,返回类型是void。
- signal函数的返回类型也是函数指针类型,该类型是void (*)(int),该函数指针指向的函数,参数是int,返回类型是void。
代码2太复杂了,能否简化呢?
可以用typedef类型重命名来解决:
》》与函数指针类型相关的内容,不能写在类型的左/右边,只能写在类型中 * 的后面。
typedef void (*pfun_t)(int);
//类型重命名
pfun_t signal(int, pfun_t);
//函数调用
然而:
typedef void (*)(int) pfun_t;
//错误
void (*)(int) signed(int, void (*)(int))
//错误
总结
本片文章我们回顾了初阶指针,又学习了两种字符指针、指针数组、数组指针及其使用、一维二维的数组传参和指针传参、函数指针。感谢大家的阅读!大家一起进步。如果文章有错误的地方,欢迎大家在评论区指正。
点赞收藏加关注,C语言学习不迷路!
相关文章:

进阶指针(一)
✨博客主页:小钱编程成长记 🎈博客专栏:进阶C语言 进阶指针(一) 0.回顾初阶指针1.字符指针1.1 相关面试题 2.数组指针3.指针数组3.1 数组指针的定义3.2 &数组名VS数组名3.3 数组指针的使用 4.数组传参和指针传参4.…...
Linux: code: name: void dev_deactivate(struct net_device *dev)
一开始看这个函数,还以为要做,网卡设备的down操作。 后来一看代码,原来这个函数是在net/sched/sch_generic.c 文件里,而且不是要做网络接口的down操作。操作的结果是:deactivate transmissions on several devices&…...

语义分割——灰度图像转伪彩色图像
目录 检验灰度图检验代码 灰度图转伪彩色图代码转换代码使用细则 示例转换结果总结 检验灰度图 制作语义分割数据集或用训练好模型测试图像时,得到的结果是灰度图像,如下: 检验代码 上面图像灰度值不是全是全为0,灰度范围在[0…...

观察级水下机器人使用系列之七机械手臂
本期是观察级水下机器人使用系列的最后一期,主要讲ROV所使用的机械臂。机械臂正式名称为5功能电动机械,型号为BE-500,由法国公司Ocean Innovation System设计的,可代替人工进行水下作业,完成海底样品采集、打捞、设备定…...
char s[]和char *s的区别,数组和指针的,堆和栈指针的一些思考
最近在学习的时候看到一个概念,数组不等价于指针,很合理但又很难理解。 例如char s[]和char *s有什么区别,前者是数组,后者是指针,个人学习成果如下: 1.char s[]和char *s的区别 char s[]: …...
Flutter快速入门学习(二)
目录 Dart介绍 一些Dart的重要概念 Dart语法学习 变量 内建类型 Number String Boolean List Set Map Symbol 函数 参数类型(可选参数,必选参数) 函数作为另一个函数的参数 匿名函数 运算符 关系运算符 类型判定运算符 赋…...

【Phoenix】phoenix实现每个Primarykey主键保留N版本数据,CDC数据记录为Changelog格式
一、背景: CDC数据中包含了,数据的变更过程。当CDC写入传统数据库最终每一个primary key下会保存一条数据。当然可以使用特殊手段保存多分记录但是显然造成了数据膨胀。 另外数据湖Hudi(0.13.1)是不支持保存所有Changelog其Compaction机制会清除所有旧版…...

阿里云服务器开放的一个新端口,重启防火墙,端口未启动
问题: 阿里云网页开放的一个新端口后,重启防火墙,端口未启动,之前配置的也都停止了。 解决: 原因可能是阿里的服务控制了,只能一个个端口开启了。把新配置新端口也单独启用。 开启80端口指令 firewall-cm…...

【PHPCUSTOM】打包PHP程序为EXE
目录 一、下载PHPCUSTOM 二、PHP网站打包 1、打开PHPCUSTOM 2、配置参数 3、生成exe文件 网上很多PHP程序打包成EXE的文章,但是都不能用,最后找到了PHPCUSTOM,使用PHPCUSTOM可以把PHP程序打包成exe。我们都知道PHP是服务端语言ÿ…...

药品咨询报告合集整理平台打包(一共36597份)【专题推荐】
<医药行业从业者必看>笔者今天分享高价值医药行业报告36500余份的获取/下载方法,报告涵盖了医药细分领域研究报告药品报告(所有上市药品)医药行业分析报告医药环境观察报告药品市场调研报告药品靶点研究报告医药白皮书;数据…...

数字化管理新革命,AI数字人CEO登场引领变革!
王一博老板乐华娱乐CEO杜华推出了她的双生数字人华华子,专门替自己直播卖货。在没有任何宣传的情况下,仅仅在短短的10分钟直播时间内,观众人数就飙升至30万人!同时,“杜华AI华华子直播”更是迅速登上了微博热搜榜。这一…...

FPGA/数字IC(芯海科技2022)面试题 2(解析版)
以下仅为学习参考(非原创),如有疑惑欢迎评论区指出! 一、单选题(共20题,每题3分,共60分) 1. D触发器:Tsetup3ns,Thold1ns,Tck2q1ns, 该D触发器最大可运行时…...

SpringMVC之JSON数据返回与异常处理机制---全方面讲解
一,JSON数据返回的理解 在Spring MVC中,当需要将数据以JSON格式返回给客户端时,可以使用ResponseBody注解或RestController注解将Controller方法的返回值直接转化为JSON格式并返回。这使得开发者可以方便地将Java对象转换为JSON,并…...
信息化发展53
数据标准化 1 、数据标准化是实现数据共享的基础。 2 、数据标准化的主要内容包括元数据标准化、数据元标准化、数据模式标准化、数据分类与编码标准化和数据标准化管理。 元数据标准化 1 、元数据是关于数据的数据( Data About Data )。其实质是用于…...
Java学习笔记——字符/字符串
在 Java 语言中,字符串都被设计成「不可变」的类型,即无法直接修改字符串的某一位字符,需要新建一个字符串实现 StringBuilder 字符 字符是用单引号括起来的单个字母,在Java中,表示字符的数据类型为char。一个字符…...

数据结构与算法基础-(1)
🌈write in front🌈 🧸大家好,我是Aileen🧸.希望你看完之后,能对你有所帮助,不足请指正!共同学习交流. 🆔本文由Aileen_0v0🧸 原创 CSDN首发🐒 如…...

华为云云耀云服务器L实例评测|轻量级应用服务器对决:基于 STREAM 深度测评华为云云耀云服务器L实例的内存性能
本文收录在专栏:#云计算入门与实践 - 华为云 专栏中,本系列博文还在更新中 相关华为云云耀云服务器L实例评测文章列表如下: 华为云云耀云服务器L实例评测 | 从零开始:云耀云服务器L实例的全面使用解析指南华为云云耀云服务器L实…...

Windows安装Neo4j
图数据库概述 图数据库是基于图论实现的一种NoSQL数据库,其数据存储结构和数据查询方式都是以图论(它以图为研究对象图论中的图是由若干给定的点及连接两点的线所构成的图形)为基础的, 图数据库主要用于存储更多的连接数据。 Neo…...
vue3开发必备核心要点
1、route/router的区别 ● $route 表示当前激活的路由的状态信息,包含了当前URL解析得到的信息,包含当前的path路径,params参数,query对象,name路径名等属性 ● r o u t e r 路由器对象( n e w 的路由器对…...

针对敏感数据的安全转录服务
即便在新冠肺炎疫情期间,继续保持了最高级别的机密性 新冠肺炎疫情带来的各种限制向所有服务提供商提出了挑战,促使提供商们想方设法采取更富想象力的新方法来满足客户的需求。澳鹏采用了一种由两种方案组成的工作机制,服务于客户机密材料的…...

《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》
引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...

CMake基础:构建流程详解
目录 1.CMake构建过程的基本流程 2.CMake构建的具体步骤 2.1.创建构建目录 2.2.使用 CMake 生成构建文件 2.3.编译和构建 2.4.清理构建文件 2.5.重新配置和构建 3.跨平台构建示例 4.工具链与交叉编译 5.CMake构建后的项目结构解析 5.1.CMake构建后的目录结构 5.2.构…...

【机器视觉】单目测距——运动结构恢复
ps:图是随便找的,为了凑个封面 前言 在前面对光流法进行进一步改进,希望将2D光流推广至3D场景流时,发现2D转3D过程中存在尺度歧义问题,需要补全摄像头拍摄图像中缺失的深度信息,否则解空间不收敛…...

基于当前项目通过npm包形式暴露公共组件
1.package.sjon文件配置 其中xh-flowable就是暴露出去的npm包名 2.创建tpyes文件夹,并新增内容 3.创建package文件夹...
C++ 基础特性深度解析
目录 引言 一、命名空间(namespace) C 中的命名空间 与 C 语言的对比 二、缺省参数 C 中的缺省参数 与 C 语言的对比 三、引用(reference) C 中的引用 与 C 语言的对比 四、inline(内联函数…...
css的定位(position)详解:相对定位 绝对定位 固定定位
在 CSS 中,元素的定位通过 position 属性控制,共有 5 种定位模式:static(静态定位)、relative(相对定位)、absolute(绝对定位)、fixed(固定定位)和…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...

UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)
UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中,UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化…...

R语言速释制剂QBD解决方案之三
本文是《Quality by Design for ANDAs: An Example for Immediate-Release Dosage Forms》第一个处方的R语言解决方案。 第一个处方研究评估原料药粒径分布、MCC/Lactose比例、崩解剂用量对制剂CQAs的影响。 第二处方研究用于理解颗粒外加硬脂酸镁和滑石粉对片剂质量和可生产…...
纯 Java 项目(非 SpringBoot)集成 Mybatis-Plus 和 Mybatis-Plus-Join
纯 Java 项目(非 SpringBoot)集成 Mybatis-Plus 和 Mybatis-Plus-Join 1、依赖1.1、依赖版本1.2、pom.xml 2、代码2.1、SqlSession 构造器2.2、MybatisPlus代码生成器2.3、获取 config.yml 配置2.3.1、config.yml2.3.2、项目配置类 2.4、ftl 模板2.4.1、…...