【C语言】—— 指针二 : 初识指针(下)

news2025/3/18 3:04:04

【C语言】——函数栈帧

一、 c o n s t const const 修饰指针

1.1、 c o n s t const const 修饰变量

  
  当我们创建一个变量后,不想被修改,该怎么办呢?这时,我们就可以请出 c o n s t const const 来修饰变量
  

int main()
{
	int m = 0;
	m = 20;//m可以被修改

	const int n = 0;
	n = 20;//n不能被修改
	return 0;
}

  
运行结果:
在这里插入图片描述
如上图,若修改被 c o n s t const const 修饰的变量,则程序报错
  
  其实, n n n 本质上还是变量,只是被 c o n s t const const 修饰后,在语法上加了限制(此时的 n n n 称作常变量),只要我们在代码中对 n n n 进行修改,就不符合语法规则,程序就会报错。
  
  虽然通过正常手段无法修改 n n n 的值,但我们可以用间接手段:通过 n n n 的地址,来修改 n n n

#include<stdio.h>

int main()
{
	const int n = 0;
	printf("n = %d\n", n);
	int* p = &n;
	*p = 20;
	printf("n = %d\n", n);
	return 0;
}

在这里插入图片描述

  但我们用 c o n s t const const 修饰变量,本质上是不希望变量被修改的,你现在通过指针把我给改了,是不是有点不讲武德。就好像我不希望你进我家,我把家门给锁,你现在翻窗进来,是不是有点不合适?
  
  所以我们应该让 p p p 拿到 n n n 的地址也修改不了 n n n ,那该怎么办呢?

  我们可以用 c o n s t const const 来修饰指针
  

1.2、 c o n s t const const 修饰指针

  

c o n s t const const 修饰指针变量的时候

  • c o n s t const const 如果放在 ∗ * 的左边,修饰的是指针所指向的内容,保证指针所指向的内容不能通过指针来修改,但是指针变量本身的内容可以改变。
  • c o n s t const const 如果放在 ∗ * 的右边,修饰的是指针变量本身,保证了指针变量本身不能被修改,但是指针指向的内容可以通过指针来改变。

注:当然, c o n s t const const 也可以在 ∗ * 两边都放的,但一般很少这么做。(你两边都不给我改,是不是太过分了)
  
通过代码来感受一下:
  
(1) c o n s t const const ∗ * 左边

#include<stdio.h>

int main()
{
	int n = 0;
	int m = 10;
	const int* p = &n;
	p = &m;
	printf("%d\n", *p);
	return 0;
}

运行结果:
在这里插入图片描述

   c o n s t const const ∗ ∗ 左边 : 指针变量本身的内容可以改变

  

参考代码:

#include<stdio.h>

int main()
{
	int n = 0;
	int m = 10;
	const int* p = &n;
	*p = 20;
	printf("%d\n", *p);
	return 0;
}

运行结果:

在这里插入图片描述
   c o n s t const const ∗ * 左边 : 指针所指向的内容不能通过指针来修改
  
(2)const 在 ∗ * 右边
  
参考代码:

#include<stdio.h>

int main()
{
	int n = 0;
	int m = 10;
	int* const p = &n;
	*p = 20;
	printf("%d\n", *p);
	return 0;
}

运行结果:
在这里插入图片描述

   c o n s t const const ∗ * 右边 : 指针指向的内容可以通过指针来改变
  
参考代码:

#include<stdio.h>

int main()
{
	int n = 0;
	int m = 10;
	int* const p = &n;
	p = &m;
	printf("%d\n", *p);
	return 0;
}

运行结果:
在这里插入图片描述

   c o n s t const const ∗ * 右边 : 指针变量本身不能被修改

  
  

二、野指针

  

概念:野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)

  

2.1野指针的成因

(1)指针未初始化
int main()
{
	int* p;//局部变量指针未初始化,默认为随机值
	*p = 20;
	return 0;
}

注:局部变量指针不初始化,默认为随机值
  

(2)指针越界访问
#include<stdio.h>

int mian()
{
	int arr[10] = { 0 };
	int* p = &arr[0];
	int i = 0;
	for (i = 0; i <= 11; i++)
	{
		//当指针指向的范围超出数组arr的范围时,p就是野指针
		*p(p++) = i;
	}
	return 0;
}

注:当指针指向超出 a r r arr arr 数组的范围时, p p p 就是野指针
  

(3)指针指向的空间释放
#include<stdio.h>

int* test()
{
	int n = 100;
	return &n;
}
int mian()
{
	int* p = test();
	printf("%d\n", *p);
	return 0;
}

注:在【C语言】——详解函数一文中,曾提到函数中形参只是实参的一份拷贝,变量 n n n 在出函数范围时已释放,这时的指针变量就是野指针。
  

2.2、如何规避野指针

(1)指针初始化
  • 明确知道指针指向哪里,就初始化一个明确的地址
  • 如果创建时,还不知道指针具体指向位置,则给指针赋值 N U L L NULL NULL

  
注: N U L L NULL NULL 是C语言中定义的一个标识符常量,本质是 00 也是地址,但该地址位于内核之中,我们用户是无法使用的,读写该地址会报错。
  

#ifndef NULL
    #ifdef __cplusplus
        #define NULL 0
    #else
        #define NULL ((void *)0)
    #endif
#endif

  

  我们可以把野指针看成野狗,很危险。给野指针赋值 NULL 相当于把野狗拴在树上,这时,你只要不靠近就不会有危险,相对安全 。

这里放张野狗图

  

(2)小心指针越界

  
  创建变量时,向内存中申请了多少空间,通过指针也只能访问这些空间。超出这些空间访问,就是越界访问,指针也就成了野指针。上述例子中,数组的越界访问就是如此。
  

(3)指针变量不再使用时,及时置 NULL,指针使用之前检查有效性

  当我们创建指针变量想访问某个区域,后期不再访问时,可以将指针变量置为 NULL
  
  使用指针变量前,我们先检查他的有效性,即检查它是否为空指针,如果为空指针,我们就不再访问
  
  还是上面野狗的例子,虽然野狗被拴起来,但是靠近他还是很危险,要和他保持一定距离才算安全。
  
  事实上,我们使用指针最合理的方式是:先判断,再使用
  

(4)避免返回局部变量的地址

  
  这一点,上述第三个例子中,不要返回函数中创建的局部变量的地址。
  
  

三、 a s s e r t assert assert 断言

  
  C语言 <assert.h> 头文件中,定义了宏 a s s e r t assert assert,那么 a s s e r t assert assert 的功能是什么呢?
  
   a s s e r t assert assert 用于在运行时程序符合规定条件,如不符合,则报错程序终止运行并给出报错信息提示
a s s e r t assert assert 常常被称作断言
  

assert(p != NULL)

  
  程序在运行到上述语句时,会先判断指针 p p p 是否为空指针,如果为空指针,则报错给出信息并终止程序运行。如果不是空指针,则程序正常运行。

   a s s e r t () assert() assert()宏接受一个表达式作为参数(该表达式不一定非要指针),该表达式为真 a s s e r t () assert() assert()宏不会产生任何作用,程序正常运行,当表达式为假, a s s e r t () assert() assert()宏就会报错,在标准错误流 s t d e r r stderr stderr 中写入一条错误信息,表示没有通过的表达式,以及包含这个表达式的文件名和行号。

那么 a s s e r t assert assert 有什么好处呢?

  • 他能自动表示文件和出问题的行号
  • 他有一种无需更改代码就开启和关闭的机制。如果想关闭 a s s e r t assert assert 断言,只需在 #include <assert.h> 前面,加上一个 N D E B U G NDEBUG NDEBUG
#define NDEBUG
#include <assert.h>

  
  而如果程序又出现问题,我们只需要将 N D E B U G NDEBUG NDEBUG 删除掉, a s s e r t () assert() assert()就能重新启动了。
  
  这时,可能还有小伙伴会问,我可以直接用 i f if if 语句来判断啊,为什么要用 a s s e r t assert assert 断言呢?
  

相比与if语句, a s s e r t assert assert 断言:

  • 出现错误,直接报错,并指明文件哪一行
  • i f if if 语句在不用时,要把他注释掉,因为让他放在那会占用内存空间

  
  当然 a s s e r t assert assert 断言也有缺点:因为引用了额外的检查,增加了程序运行时间。

  同时, a s s e r t assert assert 断言一般在 d e b u g debug debug 版本中使用,在 r e l e a s e release release 版本我们将它禁用就行,因为在 r e l e a s e release release 版本 a s s e r t assert assert 断言会直接被优化掉。
  
  

四、传值调用与传址调用

4.1、传值调用

  
  在之前的学习中,我们知道,函数传参中,形参仅仅是实参的一份临时拷贝,对形参的修改并不会改变实参的值(详情请看【C语言】——详解函数)。这种函数调用方式叫做传值调用。
  
  但如果我们想封装一个函数,让他交换两个变量的值,该怎么实现呢?这时我们就需要传址调用来实现,让我们一起来看看。
  

4.2、传址调用

  
  当我们函数传参时,将变量的地址传给函数,这种函数的调用方式叫做:传址调用。
  
我们来看下面代码:

#include<stdio.h>

void Swap(int* px, int* py)
{
	int tmp = 0;
	tmp = *px;
	*px = *py;
	*py = tmp;
}

int main()
{
	int a = 0;
	int b = 0;
	scanf("%d %d", &a, &b);
	printf("交换前: a=%d b=%d\n", a, b);
	Swap(&a, &b);
	printf("交换后: a=%d b=%d\n", a, b);

	return 0;
}

运行结果:
在这里插入图片描述

  我们可以看到,在 m a i n main main 函数中将 a a a b b b 的地址传给了 s w a p swap swap 函数, s w a p swap swap 函数通过地址,简介访问两个变量,实现了两个变量间的交换

  

结论:

  • 当我们调用函数,只需要主函数中变量的值时,可以使用传值调用
  • 当我们需要函数内部修改主函数中的值,就需要传址调用

  
  

五、二级指针

  
  在讲解二级指针之前,我们先来理清一级指针变量 ( p ) (p) p的三种关系:
  

  • p p p 中可以放 a a a 的地址
  • p p p 可以通过解引用找到 a a a
  • p p p 自己本身也有一个地址

  
如图:

在这里插入图片描述

  
  那什么又是二级指针呢?
  
  二级指针就是存放指针变量地址的变量
  

#include<stdio.h>

int main()
{
	int a = 10;
	int* pa = &a;
	int** ppa = &pa;

	return 0;
}

  
图示:

在这里插入图片描述

  
  在上述代码中, p p a ppa ppa 就是二级指针变量,变量类型 i n t int int**,怎么来理解呢?

  • 前面 i n t int int* 表示他所指向的变量为指针变量,
  • 第二颗 ∗ * 表示他一个指针变量

  
对于二级指针的运算有:

  • * p p a ppa ppa 通过对 p p a ppa ppa 中的地址进行解引用,这样找的是 p a pa pa, * p p a ppa ppa 其实访问的就是 p a pa pa
int b = 20;
*ppa = &b;//等价于 pa = &b;
  • ** p p a ppa ppa 先通过 * p p a ppa ppa 找到 p a pa pa,然后对 p a pa pa 进行解引用从操作: * p a pa pa,找到的是 a a a
**ppa = 30;
//等价于 *pa = 30;
//等价于 a = 30;

  
  

六、指针数组

6.1、指针数组的概念

  
  首先,我先来问一个问题,指针数组是指针还是数组?
  
  我们学习新知识的时候,可以用类似的旧知识来类比:整形数组是整形变量还是数组?答案很明显嘛,就是数组,同理,指针数组也是数组
  
  就像整形数组存放的元素类型是整形,指针数组存放的每个元素为指针
  

在这里插入图片描述

指针数组的每个元素是地址,又可以指向一块区域。

  

6.2、指针数组模拟二维数组

  

#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[3] = { &arr1[0],&arr2[0],&arr3[0] };

	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");
	}
}

  
运行结果:
在这里插入图片描述
  
在这里插入图片描述
  
  
   p a r r parr parr [ [ [ i i i ] ] ] 是访问 p a r r parr parr 数组的元素, p a r r parr parr [ [ [ i i i ] ] ] 找到的数组元素指向了整型一维数组, p a r r parr parr [ [ [ i i i ] ] ] [ [ [ j j j ] ] ] 就是整形一维数组中的元素。
  
  上述代码模拟出二维数组的效果,实际上并非完全是二维数组,因为二维数组在内存中是连续存储的,而上述代码每一行的存储可能差了十万八千里。
  
  
  
  


  好啦,本期关于指针就介绍到这里啦,希望本期博客能对你有所帮助,同时,如果有错误的地方请多多指正,让我们在C语言的学习路上一起进步!

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

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

相关文章

Hardness of Scheme-Switching and Comparison in FHE

参考文献&#xff1a; [AP13] Alperin-Sheriffff, J., Peikert, C.: Practical bootstrapping in quasilinear time. In: Canetti, R., Garay, J.A. (eds.) CRYPTO 2013. LNCS, vol. 8042, pp. 1–20. Springer, Heidelberg (2013). https://doi.org/10.1007/978-3-642-40041-…

微服务初识

1.认识微服务 随着互联网行业的发展&#xff0c;对服务的要求也越来越高&#xff0c;服务架构也从单体架构逐渐演变为现在流行的微服务架构。这些架构之间有怎样的差别呢&#xff1f; 1.1.单体架构 单体架构&#xff1a;将业务的所有功能集中在一个项目中开发&#xff0c;打…

Qt4 设计师自定义控件----写好qmake文件,无需额外拷贝

前言 浏览了很多帖子&#xff0c;看了很多博主的教程&#xff0c;每一个都写的很好&#xff0c;美中不足的是。Qt在每次自定义插件时都需要拷贝&#xff0c;如果能够利用qmake install拷贝功能就很完美了&#xff0c;在其他人使用的时候只要简单的几个步骤就能轻松的用起来&am…

点餐平台网站|基于springboot框架+ Mysql+Java+Tomcat的点餐平台网站设计与实现(可运行源码+数据库+设计文档+部署说明)

推荐阅读100套最新项目 最新ssmjava项目文档视频演示可运行源码分享 最新jspjava项目文档视频演示可运行源码分享 最新Spring Boot项目文档视频演示可运行源码分享 目录 前台功能效果图 管理员功能登录前台功能效果图 用户功能实现 系统功能设计 数据库E-R图设计 lunwen参…

R语言数据挖掘-关联规则挖掘(1)

一、分析目的和数据集描述 要分析的数据是美国一区域的保险费支出的历史数据。保险费用数据表的每列分别为年龄、性别、体重指数、孩子数量、是否吸烟、所在区域、保险收费。 本文的主要目的是分析在年龄、性别、体重指数、孩子数量、是否吸烟、所在区域中这些因素中&#xf…

【PTA】L1-039 古风排版(C++)

题目链接&#xff1a;L1-039 古风排版 - 团体程序设计天梯赛-练习集 (pintia.cn) 目录&#xff1a; 目录&#xff1a; 题目要求&#xff1a; 输入格式&#xff1a; 输出格式&#xff1a; 输入样例&#xff1a; 输出样例&#xff1a; 思路&#xff1a; 代码&#xff1a; 测试结…

【四 (4)数据可视化之 Ploty Express常用图表及代码实现 】

目录 文章导航一、介绍二、安装Plotly Express三、导入Plotly Express四、占比类图表1、饼图2、环形图3、堆叠条形图4、百分比堆叠条形图 五、比较排序类1、条形图2、漏斗图3、面积漏斗图 六、趋势类图表1、折线图2、多图例折线图3、分列折线图4、面积图5、多图例面积图 七、频…

解锁区块链游戏数据解决方案

作者&#xff1a;stellafootprint.network 随着区块链技术的日新月异&#xff0c;游戏行业正迎来一场革命&#xff0c;催生了区块链游戏的崛起。这一变革不仅为用户带来了全新的互动体验&#xff0c;也开辟了全新的盈利渠道。然而&#xff0c;在这一新兴领域&#xff0c;数据的…

程序人生——Java泛型和反射的使用建议

目录 引出泛型和反射建议93&#xff1a;Java的泛型是类型擦除的建议94&#xff1a;不能初始化泛型参数和数组建议95&#xff1a;强制声明泛型的实际类型 建议96&#xff1a;不同的场景使用不同的泛型通配符建议97&#xff1a;警惕泛型是不能协变和逆变的 建议98&#xff1a;建议…

安卓国产百度网盘与国外云盘软件onedrive对比

我更愿意使用国外软件公司的产品&#xff0c;而不是使用国内百度等制作的流氓软件。使用这些国产软件让我不放心&#xff0c;他们占用我的设备大量空间&#xff0c;在我的设备上推送运行各种无用的垃圾功能。瞒着我&#xff0c;做一些我不知道的事情。 百度网盘安装包大小&…

网络层_IP

传输层解决的是传输控制&#xff0c;而实际真正决定数据能否发送到对端的是网络层。网络层是有概率传输&#xff0c;而传输层是可靠性传输。所以传输层网络层就可以做到将数据可靠发送到对端。网络层的常见协议有&#xff1a;IP、ICMP等&#xff0c;其中最重要的是IP协议&#…

HTML、XHTML和HTML5系列对比

目录 HTML HTML的优点&#xff1a; HTML的缺点&#xff1a; 应用场景&#xff1a; XHTML XHTML的优点&#xff1a; XHTML的缺点&#xff1a; 应用场景&#xff1a; HTML5 HTML5的优点&#xff1a; HTML5的缺点&#xff1a; 应用场景&#xff1a; 回首发现&#xff0…

Flutter Inspector 视图调试工具突然不能用了

The embedded browser failed to load. Error: JCEF is not supported in this env or failed to initialize 1、在 Android Studio 的 Help 菜单中&#xff0c;找到 Find Action 2、搜索 boot runtime&#xff0c;找到「Choose Boot Java Runtime for the IDE」选项 3、在「…

串行通信——IIC总结

一.什么是IIC&#xff1f; IIC&#xff08;Inter-Integrated Circuit&#xff09;也称I2C&#xff0c;中文叫集成电路总线。是一个多主从的串行总线&#xff0c;由飞利浦公司发明的通讯总线&#xff0c;属于半双工同步传输类总线&#xff0c;仅由两条线就能完成多机通讯&#…

电竞游戏行业有哪些媒体资源?活动发布会如何宣传?

传媒如春雨&#xff0c;润物细无声&#xff0c;大家好&#xff0c;我是51媒体网胡老师。 电竞游戏行业的媒体资源主要包括&#xff1a;游戏门户网站、综合资讯网站、社交媒体平台、电视和网络直播等。 在电竞游戏行业中&#xff0c;媒体资源是丰富多样的。游戏门户网站如游民…

sql中使用collection返回集合数据

今天在写一个接口时&#xff0c;有两级目录&#xff08;父子关系&#xff09;&#xff0c;接口需要把两级数据以嵌套的形式返回给前端。我这个新手菜鸟一上来就查询两次sql&#xff0c;然后业务中处理嵌套关系&#xff0c;事实这种方法也能达到目的。但主管PR代码时&#xff0c…

【自动驾驶可视化工具】

自动驾驶可视化工具 自动驾驶可视化工具1.百度Apollo的Dreamview:2.Cruise的Worldview:3.Uber的AVS:4.Fglovex Studio: 自动驾驶可视化工具 介绍一下当前主流的自动驾驶可视化工具。 1.百度Apollo的Dreamview: Dreamview是百度Apollo平台开发的一种可视化工具&#xff0c;用…

华为配置中心AP内漫游实验

华为配置中心AP内漫游示例 组网图形 图1 配置中心AP内漫游组网图 配置流程组网需求配置思路数据规划配置注意事项操作步骤配置文件 配置流程 WLAN不同的特性和功能需要在不同类型的模板下进行配置和维护&#xff0c;这些模板统称为WLAN模板&#xff0c;如域管理模板、射频模…

K8S日志收集方案-EFK部署

EFK架构工作流程 部署说明 ECK (Elastic Cloud on Kubernetes)&#xff1a;2.7 Kubernetes&#xff1a;1.23.0 文件准备 crds.yaml 下载地址&#xff1a;https://download.elastic.co/downloads/eck/2.7.0/crds.yaml operator.yaml 下载地址&#xff1a;https://download.e…

javaweb-maven+HTTP协议+Tomcat+SpringBoot入门+请求+响应+分层解耦

Maven IDEA集成Maven 依赖管理 依赖配置 maven是插件完成对应的工作的~ 哇哇哇maven看完啦~~~~~~ Spring.io Springboot是Spring家族的子项目&#xff0c;可以帮助我们非常快速地构建应用程序&#xff0c;简化开发&#xff0c;提高效率。 RestController请…