深入浅出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.mzph.cn/diannao/49081.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

便携式自动气象站:科技赋能气象观测

便携式自动气象站&#xff0c;顾名思义&#xff0c;就是一款集成了多种气象传感器&#xff0c;能够自动进行气象观测和数据记录的设备。它体积小巧、重量轻&#xff0c;便于携带和快速部署&#xff0c;可以在各种环境下进行气象数据的实时监测。同时&#xff0c;通过内置的无线…

版本更新 | Orillusion 0.8发布,与大家同在!

过了这么久&#xff0c;我们Orillusion引擎的大版本更新终于来啦&#xff01; 这次的版本发布&#xff0c;大部分是更新了引擎底层能力&#xff0c;有兴趣的小伙伴可以直接查看&#xff1a; &#x1f517; https://github.com/Orillusion/orillusion 其实面对社区的小伙伴&…

应对爬虫过程中代理IP掉线的实用指南

当代理IP在爬虫中频繁掉线时&#xff0c;我们先要了解出现问题的可能原因&#xff0c;这不仅限于技术性因素&#xff0c;还涉及操作策略和环境因素。只有在找到具体原因后&#xff0c;才能针对问题类型从源头解决IP掉线问题。 一、问题原因&#xff1a; 1. 代理IP质量问题导致…

Python将字典转换为DataFrame的实战代码

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

AWS监控工具,监控性能指标

执行AWS监视是为了跟踪在AWS环境中主动运行的应用程序工作负载和资源&#xff0c;AWS监视器跟踪各种AWS云指标&#xff0c;以帮助提高在其上运行的应用程序的整体性能。 借助阈值突破警报系统&#xff0c;AWS应用程序监控在识别性能瓶颈来源方面起着至关重要的作用&#xff0c…

力扣高频SQL 50题(基础版)第五题

文章目录 力扣高频SQL 50题&#xff08;基础版&#xff09;第五题1683. 无效的推文题目说明&#xff1a;思路分析&#xff1a;实现过程&#xff1a;结果截图&#xff1a; 力扣高频SQL 50题&#xff08;基础版&#xff09;第五题 1683. 无效的推文 题目说明&#xff1a; 表&a…

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

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

Xlua原理 二

一已经介绍了初步的lua与C#通信的原理&#xff0c;和xlua的LuaEnv的初始化内容。 这边介绍下Wrap文件。 一.Wrap介绍 导入xlua后可以看到会多出上图菜单。 点击后生成一堆wrap文件&#xff0c;这些文件是lua调用C#时进行映射查找用的中间代码。这样就不需要去反射调用节约性…

ubuntu安装mysql8.0

文章目录 ubuntu版本安装修改密码取消root跳过密码验证 ubuntu版本 22.04 安装 更新软件包列表 sudo apt update安装 MySQL 8.0 服务器 sudo apt install mysql-server在安装过程中&#xff0c;系统可能会提示您设置 root 用户的密码&#xff0c;请务必牢记您设置的密码。…

【中项】系统集成项目管理工程师-第4章 信息系统架构-4.3应用架构

前言&#xff1a;系统集成项目管理工程师专业&#xff0c;现分享一些教材知识点。觉得文章还不错的喜欢点赞收藏的同时帮忙点点关注。 软考同样是国家人社部和工信部组织的国家级考试&#xff0c;全称为“全国计算机与软件专业技术资格&#xff08;水平&#xff09;考试”&…

linux中RocketMQ安装(单机版)及springboot中的使用

文章目录 一、安装1.1、下载RocketMQ1.2、将下载包上传到linux中&#xff0c;然后解压1.3、修改runserver.sh的jvm参数大小&#xff08;根据自己服务器配置来修改&#xff09;1.4、启动mqnamesrv &#xff08;类似于注册中心&#xff09;1.5、修改runbroker.sh的jvm参数大小&am…

Kafka Producer之事务性

文章目录 1. 跨会话幂等性失效2. 开启事务3. 事务流程原理 事务性可以防止跨会话幂等性失效&#xff0c;同时也可以保证单个生产者的指定数据&#xff0c;要么全部成功要么全部失败&#xff0c;不限分区。不可以多个生产者共用相同的事务ID。 1. 跨会话幂等性失效 幂等性开启…

Spring MVC笔记

Java 版本: JDK17 Eclipse: eclipse-jee-2023-12-R-win32-x86_64.zip Tomcat 10 JDK17采用springframework 6 *必须考虑兼容性问题,所以JDK 和spring framework不要乱搭配 初步创建Maven Project 安装包 修改poem.xml <dependency><groupId>org.springframework…

Linux中tomcat下载教程

一.安装tomcat 1.安装 EPEL 仓库&#xff1a; sudo yum install epel-release2.安装 Tomcat&#xff1a; sudo yum install tomcat3.启动 Tomcat 服务&#xff1a; sudo systemctl start tomcat4.启用 Tomcat 服务开机启动&#xff1a; sudo systemctl enable tomcat5.检查…

大语言模型-Bert-Bidirectional Encoder Representation from Transformers

一、背景信息&#xff1a; Bert是2018年10月由Google AI研究院提出的一种预训练模型。 主要用于自然语言处理&#xff08;NLP&#xff09;任务&#xff0c;特别是机器阅读理、文本分类、序列标注等任务。 BERT的网络架构使用的是多层Transformer结构&#xff0c;有效的解决了长…

计算机网络知识-面试点1

1. 三握四挥 定义&#xff1a; 在计算机网络中&#xff0c;特别是TCP/IP协议中&#xff0c;“三握”指的是三次握手&#xff08;Three-way Handshake&#xff09;&#xff0c;而“四挥”则指的是四次挥手&#xff08;Four-way Handshake&#xff09;。这两个过程分别用于TCP连接…

LangChain的使用详解

一、 概念介绍 1.1 Langchain 是什么&#xff1f; 官方定义是&#xff1a;LangChain是一个强大的框架&#xff0c;旨在帮助开发人员使用语言模型构建端到端的应用程序&#xff0c;它提供了一套工具、组件和接口&#xff0c;可简化创建由大型语言模型 (LLM) 和聊天模型提供…

Qt实战:专栏内容介绍及目录

1、专栏介绍 Qt相比Visual Studio (VS) 的优势主要体现在跨平台能力、‌丰富的功能、‌高性能、‌现代UI设计、‌社区支持和企业支持等方面。‌ 跨平台能力&#xff1a;‌Qt 允许应用程序在多个操作系统上编译和运行&#xff0c;‌无需为每个平台编写特定的代码&#xff0c;‌…

构建高效园区导览系统:基于3DGIS与物联网技术的实现方案

园区导航的挑战与机遇 在现代化的大型园区中&#xff0c;随着面积的不断扩张和布局的日益复杂&#xff0c;传统的纸质地图已难以满足日益增长的导航需求。每栋楼、每层楼都有着不同的办公室&#xff0c;不同的业务。这种低效的寻路过程不仅影响了客户的来访体验&#xff0c;也…

SSD基本架构与工作原理

SSD的核心由一个或多核心的CPU控制器、DRAM缓存以及多个NAND闪存芯片组成。CPU控制器负责管理所有读写操作&#xff0c;并通过DRAM缓存存储映射表等元数据&#xff0c;以加速寻址过程。 NAND闪存则是数据存储的实际介质&#xff0c;其组织结构从大到小依次为通道&#xff08;包…