深入浅出C语言指针(进阶篇)

深入浅出C语言指针(基础篇)

深入浅出C语言指针(进阶篇)

目录

引言

一、指针和数组

1.数组名的理解

2.指针访问数组 

3.一维数组传参的本质

二、二级指针

1.二级指针的概念

2.二级指针的内存表示

3.二级指针的解引用

三、字符指针

1.指针指向单个字符

2.指针指向字符串

3.一道面试题

四、指针数组

1.指针数组的概念

 2.初始化指针数组

3.指针数组模拟二维数组

五、数组指针

1.数组指针的概念

2.初始化数组指针

3.二维数组传参的本质

六、函数指针

1.函数指针的概念

2.初始化函数指针

3.函数指针的应用

七、函数指针数组

1.函数指针数组的概念

2.初始化函数指针数组

 3.函数指针数组的应用

 总结


引言

在C语言中,指针是至关重要的一部分,掌握指针的用法对于编写高效、简洁的代码具有极大帮助。本文将带您深入了解C语言指针的高级用法,助您迈向编程高手之路。

一、指针和数组

1.数组名的理解

请看下面一段代码:

#include <stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };printf("arr =   %p\n", arr);printf("&arr[0]=%p\n", &arr[0]);return 0;
}

运行结果如下: 

我们发现数组名和数组⾸元素的地址打印出的结果⼀模⼀样, 数组名就是数组⾸元素的地址

 两个特例

sizeof(数组名) ,sizeof中单独放数组名,这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩,
单位是字节
&数组名 ,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素的地址是有区别的)
除此之外,任何地⽅使⽤数组名,数组名都表⽰⾸元素的地址。

2.指针访问数组 

我们搞清楚数组名就是数组首元素的地址之后,那么这样写代码就是可行的:

int arr[10] = {1,2,3,4,5,6,7,8,9,0};
int *p = arr;//p存放的是数组首元素的地址

当我们把数组名当成地址存放到指针中,就可以用指针访问数组了:

#include <stdio.h>
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };int len = sizeof(arr) / sizeof(arr[0]);int* p = arr;//使用指针来访问数组int i = 0;for (i = 0; i < len; i++){printf("%d ", *(p + i));//这里*(p+i)等价于arr[i]}return 0;
}

可以看到成功访问了数组每个元素: 

3.一维数组传参的本质

再请看下面代码:

#include <stdio.h>
void test1(int arr[])
{printf("%d\n",sizeof(arr));
}
void test2(int *arr)
{printf("%d\n", sizeof(arr));
}
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };test1(arr);test2(arr);return 0;
}

输出结果如下:

上面我们学习了:数组名是数组⾸元素的地址;那么在数组传参的时候,传递的是数组名,也就是说本质上 数组传参本质上传递的是数组⾸元素的地址。
所以函数形参的部分理论上应该使⽤指针变量来接收⾸元素的地址。那么在函数内部我们写
sizeof(arr) 计算的是⼀个地址的⼤⼩(单位字节)⽽不是数组的⼤⼩(单位字节)
由此我们可以总结

数组传参的本质是传递了数组首元素的地址,所以形参访问的数组和实参访问的数组是同一个数组。(所以形参的数组并不会再开辟空间,故也可以省略数组大小只写成arr[])
⼀维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式。

二、二级指针

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪⾥呢?

答案是二级指针

1.二级指针的概念

在C语言中,指针是一个变量,其值为另一个变量的地址。而二级指针是一个指向指针的指针,即它的值是另一个指针的地址。

二级指针的定义可以表示为:

int a = 10;        // 声明一个整型变量a
int *p = &a;       // 声明一个指向整型变量a的指针p
int **pp = &p;     // 声明一个指向指针p的二级指针pp

在这个例子中,a的地址存放在p中,p的地址存放在pp中。p是一级指针,存储a的地址;pp 是一个二级指针,它存储的是指针 p 的地址。

2.二级指针的内存表示

3.二级指针的解引用

要获取二级指针所指向的指针所指向的变量的值,需要进行两次解引用:

#include <stdio.h>
int main() {int a = 10;int* p = &a;int** pp = &p;// 通过二级指针修改a的值**pp = 20;printf("a = %d\n", a);return 0;
}

这里的** pp 首先通过* pp 获取 p 的值(即 a 的地址)然后通过* 解引用这个地址,从而修改 a 的值。

三、字符指针

1.指针指向单个字符

这是一般的使用方法,这里进阶篇我们就不过多赘述:

int main()
{char ch = 'w';char *pc = &ch;*pc = 'w';return 0;
}

2.指针指向字符串

#include<stdio.h>
int main()
{const char* pstr = "abcdef";//这里是把一个字符串放到pstr指针变量里了吗?printf("%s\n", pstr);//%s打印字符串后面需提供地址return 0;
}

代码 const char* pstr = "abcdef"; 特别容易让人以为是把字符串abcdef放到字符指针pstr里了,但是本质是把字符串abcdef的首字符a的地址放到了pstr中。由于字符串是连续存放的,故也可以通过通过指针访问到整个字符串。

3.一道面试题

《剑指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;
} 

答案如下:

我们来分析一下:这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当几个指针指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4相同。

总结如下:
.str1==str2比较的是地址,而不是内容。
每个数组对象都会分配属于自己的存储空间,地址各不相同。
指针变量并不分配存储区,而是指向常量字符串所在的静态存储区,由于指向的是同一个常量,所以str3=str4。

四、指针数组

1.指针数组的概念

指针数组是指针还是数组?
我们类⽐⼀下,整型数组,是存放整型的数组,字符数组是存放字符的数组。
那指针数组呢?是存放指针的数组

指针数组:在C语言中,指针数组是一种特殊类型的数组,指针数组常用于存储一系列的地址,这些地址可以是变量的地址、数组元素的地址或者其他指针的地址。简单来说指针数组是一个数组,其中的每一个元素都是指向某种数据类型的指针。

声明数据类型*指针数组名[数组长度]

int *arr[5];

 2.初始化指针数组

在声明时进行初始化:

int a=1,b=2,c=3,d=4,e=5;
int *arr[5]={&a,&b,&c,&d,&e};

 也可以先声明一个指针数组,然后在后续代码中赋值:

int a=1,b=2,c=3,d=4,e=5;
int *arr[5];
arr[0]=&a;
arr[1]=&b;
arr[2]=&c;
arr[3]=&d;
arr[4]=&e;

3.指针数组模拟二维数组

#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*的,就可以存放在parr数组中int* parr[3] = { arr1, arr2, arr3 };int i = 0;int j = 0;for (i = 0; i < 3; i++){for (j = 0; j < 5; j++){printf("%d ", parr[i][j]);}printf("\n");}return 0;
}

parr[i]是访问parr数组的元素,parr[i]找到的数组元素指向了整型⼀维数组,parr[i][j]就是整型⼀维数组中的元素。上述的代码模拟出⼆维数组的效果,实际上并⾮完全是⼆维数组,因为每⼀⾏并⾮是连续的。

五、数组指针

1.数组指针的概念

数组指针是指针?还是数组?
答案是:指针。
我们已经熟悉:
整形指针: int * pi ; 能够指向整形数据的指针。
浮点型指针: float * pf ; 能够指向浮点型数据的指针。
那数组指针应该是:能够指向数组的指针。
数组指针:在C语言中,数组指针是一种特殊的指针类型,它指向的是一个数组。 更具体地说,数组指针是一个指针,其指向的是包含特定数量元素的数组。数组指针与指向单个变量的指针不同,它指向的是一个数组的首元素, 但是它的解引用操作会返回整个数组,而不是单个元素
声明:数据类型(*指针名)[数组长度]
int(*ptr)[5];

2.初始化数组指针

数组指针可以通过取数组的地址来初始化:

int arr[5] = {1, 2, 3, 4, 5};
int (*ptr)[5] = &arr; // 初始化数组指针,使其指向数组arr|    |    ||    |    ||    |    ptr指向数组的元素个数|    ptr是数组指针变量名ptr指向的数组的元素类

在这里,&arr 表示取数组 arr 的地址(而不是首元素的地址),这个地址就是数组指针 ptr 的初始值。

3.二维数组传参的本质

 有了数组指针的理解,我们就能够讲⼀下⼆维数组传参的本质了。

过去我们有⼀个⼆维数组的需要传参给⼀个函数的时候,我们是这样写的:
#include <stdio.h>
void test(int a[3][5], int r, int c)
{int i = 0;int j = 0; for (i = 0; i < r; i++){for (j = 0; j < c; j++){printf("%d ", a[i][j]);}printf("\n");}
}
int main()
{int arr[3][5] = { {1,2,3,4,5}, {2,3,4,5,6},{3,4,5,6,7} };test(arr, 3, 5);return 0;
} 
这⾥实参是⼆维数组,形参也写成⼆维数组的形式,那还有什么其他的写法吗?
⾸先我们再次理解⼀下⼆维数组,⼆维数组起始可以看做是每个元素是⼀维数组的数组,也就是⼆维数组的每个元素是⼀个⼀维数组。那么⼆维数组的⾸元素就是第⼀⾏,是个⼀维数组。
如下图:

所以,根据数组名是数组⾸元素的地址这个规则, ⼆维数组的数组名表⽰的就是第⼀⾏的地址,是
维数组的地址,类型是数组指针类型 。根据上⾯的例⼦,第⼀⾏的⼀维数组的类型就是 int [5] , 所以第⼀⾏的地址的类
型就是数组指针类型 int(*)[5] 。那就意味着⼆维数组传参本质上也是传递了地址,传递的是第⼀
⾏这个⼀维数组的地址,那么形参也是可以写成指针形式的。如下:
#include <stdio.h>
void test(int(*p)[5], int r, int c)
{int i = 0;int j = 0;for (i = 0; i < r; i++){for (j = 0; j < c; j++){printf("%d ", *(*(p + i) + j));}printf("\n");}
}
int main()
{int arr[3][5] = { {1,2,3,4,5}, {2,3,4,5,6},{3,4,5,6,7} };test(arr, 3, 5);return 0;
} 

运行结果:

总结:⼆维数组传参,形参的部分可以写成数组,也可以写成指针

六、函数指针

1.函数指针的概念

函数指针:在C语言中,函数指针是一种特殊类型的指针,它存储的是函数的地址,而不是变量的地址。通过函数指针,你可以间接地调用函数,类似于通过普通指针间接访问变量。

声明返回类型 (*函数指针名)(参数类型1, 参数类型2, ...);

这里的返回类型是函数返回的类型,函数指针名是你给这个指针起的名字,而参数类型1, 参数类型2, ... 是函数的参数列表。

int(*fubPtr)(int,int);

2.初始化函数指针

函数指针可以通过取函数的地址来初始化:

int add(int a, int b) {return a + b;
}
int (*funcPtr)(int, int) = &add; // 初始化函数指针,使其指向add函数|    |       ----------          //也可以写成int (*funcPtr)(int, int)=add;|    |           ||    |         funPtr指向函数的参数类型和个数的交代|   函数指针变量名funPtr指向函数的返回类型

在这里,&add 表示取函数 add 的地址,这个地址就是函数指针 funcPtr 的初始值。

(这里add和&add是一样的,函数名和&函数名都表示函数地址,二者没有区别)

3.函数指针的应用

通过函数指针调⽤指针指向的函数
#include <stdio.h>
int Add(int x, int y)
{return x + y;
}
int main()
{int(*pf3)(int, int) = Add;printf("%d\n", Add(1, 2));//通过函数名调用printf("%d\n", (*pf3)(2, 3));//通过函数指针调用printf("%d\n", pf3(3, 5));//由于pf存储的是函数的地址,也可以通过函数地址直接调用,故也可以不写*号return 0;
}

运行结果如下:

七、函数指针数组

1.函数指针数组的概念

函数指针数组:在C语言中,函数指针数组是一个数组,其元素是函数指针。这意味着函数指针数组中的每个元素都是一个指针,指向一个函数。函数指针数组可以用来存储多个函数的地址,并允许你在运行时根据需要选择其中一个函数来执行。

声明:返回类型 (*数组名)[数组长度];

这里的 返回类型 是函数返回的类型,数组名 是数组的名字,而 数组长度 是一个常量表达式,表示数组中函数指针的数量。

int(*arr[5])(int,int);//在这个声明中,arr是一个数组,它包含5个指向整型函数的指针。

2.初始化函数指针数组

函数指针数组可以通过取函数的地址来初始化:

int add(int a, int b) {return a + b;
}int subtract(int a, int b) {return a - b;
}int (*funcArray[2])(int, int) = {add, subtract}; // 初始化函数指针数组

在这里,funcArray 是一个数组,它包含两个指向整型函数的指针,分别指向 add 和 subtract 函数。

 3.函数指针数组的应用

转移表

#define _CRT_SECURE_NO_WARNINGS
#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;int ret = 0;int(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //转移表do {printf("**********计算器**********\n");printf("******1.加法  2.减法******\n");printf("******3.乘法  4.除法******\n");printf("*********0.退出**********\n");printf("请选择:\n");scanf("%d", &input);if (input >= 1 && input <= 4){printf("输入需要操作的两个数\n");scanf("%d%d", &x, &y);ret = (*p[input])(x, y);printf("结果为%d\n", ret);}else if (input = 0){break;}else{printf("输入有误,请重新输入!\n");}} while (input);
}

运行如下:

 总结

通过本文的学习,相信你对C语言指针有了更深入的理解,能够更好地运用指针来编写高效的C语言程序。加油!祝你更上一层楼!!!

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

【目标检测】Anaconda+PyTorch配置

前言 本文主要介绍在windows系统上的Anaconda、PyTorch关键步骤安装&#xff0c;为使用yolo所需的环境配置完善。同时也算是记录下我的配置流程&#xff0c;为以后用到的时候能笔记查阅。 Anaconda 软件安装 Anaconda官网&#xff1a;https://www.anaconda.com/ 另外&#…

Golang | Leetcode Golang题解之第278题第一个错误的版本

题目&#xff1a; 题解&#xff1a; func firstBadVersion(n int) int {return sort.Search(n, func(version int) bool { return isBadVersion(version) }) }

Elasticsearch基础(六):使用Kibana Lens进行数据可视化

文章目录 使用Kibana Lens进行数据可视化 一、进入Kibana Lens 二、基础可视化 1、指标可视化 2、垂直堆积条形图 3、表格 三、高级可视化 1、多图层和索引 2、子桶 3、树状图 使用Kibana Lens进行数据可视化 一、进入Kibana Lens 在Kibana主页&#xff0c;单击页面…

vxe-table——实现切换页码时排序状态的回显问题(ant-design+elementUi中table排序不同时回显的bug)——js技能提升

之前写的后台管理系统&#xff0c;都是用的antdelement&#xff0c;table组件中的【排序】问题是有一定的缺陷的。 想要实现的效果&#xff1a; antv——table组件一次只支持一个参数的排序 如下图&#xff1a; 就算是可以自行将排序字段拼接到列表接口的入参中&#xff0c…

Druid【基础 01】是什么+主要特点+设计原则+架构+数据结构(简单入门Druid)

Druid入门 1. 是什么2. 主要特点3. 三个设计原则4. Architecture 架构5. 数据结构5.1 DataSource 结构5.2 Segment 结构 Druid 非中文官网&#xff0c;内容不少且介绍的挺详细的&#xff0c;需要英文阅读能力或者翻译工具进行辅助。 1. 是什么 先看看官网怎么说&#xff1a; A…

请你谈谈:spring bean的生命周期 - 阶段5:BeanPostProcessor前置处理-自定义初始化逻辑-BeanPostProcess后置处理

BeanPostProcessor的postProcessBeforeInitialization方法是在bean的依赖注入&#xff08;即属性填充&#xff09;完成后&#xff0c;但在bean的初始化回调&#xff08;如PostConstruct注解的方法或InitializingBean接口的afterPropertiesSet方法&#xff09;之前被调用的。 具…

证书上的服务器名错误解决方法

方法 win r &#xff0c;输入mmc 点击文件——>添加/删除管理单元 找到证书——> 添加 根据自己的存放选择存放位置 点击控制台根节点——> 受信任的根证书颁发机构——>导入 若还出现问题&#xff0c;则参考https://blog.csdn.net/mm120138687/article/details/…

立创梁山派--移植开源的SFUD万能的串行 Flash 通用驱动库

SFUD是什么 关于SFUD库的介绍&#xff0c;其开源链接(gitee,github)已经详细的阐述了. 这里是截取自它的一部分介绍&#xff1a; SFUD 是一款开源的串行 SPI Flash 通用驱动库。由于现有市面的串行 Flash 种类居多&#xff0c;各个 Flash 的规格及命令存在差异&#xff0c; SF…

Apache Tomcat文件包含漏洞复现(详细教程)

1.漏洞原理 Tomcat 服务器是一个免费的开放源代码的Web 应用服务器&#xff0c;其安装后会默认开启ajp连接器&#xff0c;方便与其他web服务器通过ajp协议进行交互。属于轻量级应用服务器&#xff0c;在中小型系统和并发访问用户不是很多的场合下被普遍使用&#xff0c;是开发和…

【接口自动化_07课_Pytest+Excel+Allure完整框架集成_下】

目标&#xff1a;优化框架场景 1. 生成对应的接口关联【重点】 2. 优化URL基础路径封装【理解】 3. 利用PySQL操作数据库应用【理解】--- 怎么用python连接数据库、mysql 4. 通过数据库进行数据库断言【重点】 5. 通过数据库进行关联操作【重点】 一、接口关联&#xff1a…

MSP430M03507最小系统板的keil环境搭配,用keil编辑ti单片机

转载自嘉立创MSP430M03507开发手册 这篇文章只是因为我的keil版本与嘉立创的不一样&#xff0c;所以添加了我自己遇到的问题解析 先说说为什么要用keil编辑&#xff0c;因为ti单片机自己的ccs编译环境需要对应仿真器&#xff0c;那个加芯片都240了&#xff0c;哪有那么多钱买…

node.js中nodemon : 无法加载和使用问题,这是由于windows安全策略影起的按如下操作即可

1、用管理员权限打开vscode 2、文件终端中打开&#xff0c;输入 Set-ExecutionPolicy -Scope CurrentUser 3、再输入RemoteSigned 4、使用get-ExecutionPolicy查看权限&#xff0c;可以看到变为了RemoteSigned 重启问题解决

MySQL面试索引篇

1、什么是索引&#xff1f; 作为一个数据库&#xff0c;首要任务就是把数据存储好&#xff0c;并快速查询出用户需要的数据&#xff0c;而索引就相当于图书的目录一样&#xff0c;是一种用于快速查询和检索数据的数据结构&#xff0c;其本质可以看成是一种排序好的数据结构。 …

TypeScript 教程(九):类型声明文件与异步编程

目录 前言回顾装饰器与高级类型操控1. 类型声明文件a. 什么是类型声明文件&#xff08;.d.ts&#xff09;b. 编写和使用类型声明文件 2. 异步编程a. Promise 类型b. async/awaitc. 异步迭代器 3. 并行执行与错误处理a. Promise.allb. Promise.racec. 错误处理 结语 前言 在前几…

华为云.云日志服务LTS及其基本使用

云计算 云日志服务LTS及其基本使用 - 文章信息 - Author: 李俊才 (jcLee95) Visit me at CSDN: https://jclee95.blog.csdn.netMy WebSite&#xff1a;http://thispage.tech/Email: 291148484163.com. Shenzhen ChinaAddress of this article:https://blog.csdn.net/qq_28550…

数学建模(7)——Logistic模型

一、马尔萨斯人口模型 import numpy as np import matplotlib.pyplot as plt# 初始人口 N0 100 # 人口增长率 r 0.02 # 时间段&#xff08;年&#xff09; t np.linspace(0, 200, 200)# 马尔萨斯人口模型 N N0 * np.exp(r * t)# 绘图 plt.plot(t, N, labelPopulation) plt.…

图片转pdf的软件有哪些?这几种转换工具了解下

在日常的办公学习中&#xff0c;图片转PDF的需求愈发普遍。不论是工作汇报、学习笔记还是生活点滴&#xff0c;我们都希望将重要的图片内容整理成易于查阅的PDF格式。那么&#xff0c;有哪些软件可以做到将图片转换成PDF格式呢&#xff1f;给大家介绍5种简单好用的转换方法&…

Linux第五节课(权限02)

1、Linux下的用户分类 root&#xff1a;超级用户普通用户&#xff1a;通过root新建的用户&#xff0c;adduser root不受权限约束&#xff1b;普通用户受权限约束&#xff1b; Linux系统中&#xff0c;所有用户都需要有密码&#xff0c;无论是root还是其他&#xff0c;即便是…

SpringBoot+ Sharding Sphere 轻松实现数据库字段加解密

一、介绍 在实际的软件系统开发过程中&#xff0c;由于业务的需求&#xff0c;在代码层面实现数据的脱敏还是远远不够的&#xff0c;往往还需要在数据库层面针对某些关键性的敏感信息&#xff0c;例如&#xff1a;身份证号、银行卡号、手机号、工资等信息进行加密存储&#xf…

优选算法之二分查找(上)

目录 一、二分查找 1.题目链接&#xff1a;704. 二分查找 2.题目描述&#xff1a; 3.算法流程&#xff1a; 4.算法代码&#xff1a; 二、在排序数组中查找元素的第一个和最后一个位置 1.题目链接&#xff1a;34. 在排序数组中查找元素的第一个和最后一个位置 2.题目描述…