C语言——从头开始——深入理解指针(1)
一.内存和地址
我们知道计算上CPU(中央处理器)在处理数据的时候,是通过地址总线把需要的数据从内存中读取的,后通过数据总线把处理后的数据放回内存中。如下图所示:
计算机把内存划分为⼀个个的内存单元,每个内存单元的大小取1个字节( 1个字节(Byte)=8个比特位(bit)), 再对每个内存单元进行编号处理,这样就可以高效管理内存。
而在计算机中我们把内存单元的编号也称为地址。C语⾔中给地址起了新的名字叫:指针。
所以通过以上可总结:内存单元的编号 == 地址 == 指针 可见指针就是地址,地址就是指针。指针的作用就是访问内存的
二.指针变量和地址
理解了内存和地址的关系,就可以理解,在C语⾔中创建变量其实就是向内存申请空间,如下:
上述的代码就是创建了整型变量a,在内存中申请4个字节,⽤于存放整数9,其中每个字节都有地址。
那我们如何能得到 a 的地址呢? 这⾥就得学习⼀个操作符:&——取地址操作符(单目操作符)
上图所示,会打印出:0x006FFD70 (因为&a取出的是a所占4个字节中地址较小的字节的地址)
以上我们通过取地址操作符(&)拿到的地址是⼀个数值,⽐如:0x006FFD70,这个数值有时候也是需要 存储起来,⽅便后期再使⽤,那我们把这样的地址值存放在哪⾥呢?答:指针变量中。如下:
以上p为指针变量。p左边写的是 int* , * 是在说明p是指针变量,⽽前⾯的 int 是在说明p指向的是整型(int) 类型的对象。(指针变量也是⼀种变量,这种变量就是⽤来存放地址的,存放在指针变量中的值都会理解为地址。 )
我们将地址保存起来,未来是要使⽤的,那怎么使⽤呢?
C语言中,我们只要拿到了地址(指针),就可以通过地址(指针)找到地址(指针)
指向的对象,则我们必须学习⼀个操作符:* —— 解引⽤操作符
*p 的意思就是通过 p 中存放的地址,找到指向的空间, *p其实就是a变量了;所以*p = 2,这个操作符就是把a改成了2。
这里有同学肯定在想,如果⽬的就是把a改成2的话,写成 a = 2; 不就完了吗? 为啥⾮要使⽤指针呢? 其实这⾥是把a的修改交给了p来操作,这样对a的修改,就多了⼀种的途径,写代码就会更加灵活!
指针变量的大小:
由于32位机器有32根地址总线,每根地址线出来的电信号转换成数字信号后是1或者0,那我们把32根地址线产⽣的2进制序列当做⼀个地址,那么⼀个地址就是32个bit位,需要4个字节才能存储
则指针变量的大小就是4个字节。如下图所示:
同理64位机器,有64根地址线,⼀个地址就是64个⼆进制位组成的⼆进制序列,存储起来就需要
8个字节的空间,则指针变量的大小就是8个字节。 如下图所示:
由上述总结可得:指针变量的大小取决于地址的大小
- 32位平台下地址是32个bit位(即4个字节)
- 64位平台下地址是64个bit位(即8个字节)
- 指针变量的⼤⼩和类型是⽆关的,只要指针类型的变量,在相同的平台下,⼤⼩都是相同的。
三. 指针变量类型的意义
- 指针的解引用:
以上代码通过调试我们可以看到,其只是将n的第⼀个字节改为0
则可得出结论:指针的类型决定了,对指针解引⽤的时候有多⼤的权限(⼀次能操作⼏个字节)
⽐如: char* 的指针解引⽤就只能访问⼀个字节,而 int* 的指针的解引⽤就能访问四个字节
- 指针 +- 整数:
观察以上代码可总结出:指针变量类型决定了指针进行加1,减1的时候,能偏移几个字节(一次能走多远 )
- void* 指针(泛型指针): 无具体类型的指针
这种类型的指针可以⽤来接受任意类型地址。但是也有局限性, void* 类型的指针不能直接
进行指针的 +- 整数和解引⽤的运算。如下图:
注:void* 类型的指针不能直接进行指针的解引用 和 +- 整数的运算
那么 void* 类型的指针到底有什么⽤呢?
⼀般 void* 类型的指针是使⽤在函数参数的部分,⽤来接收不同类型数据的地址,这样的设计可以
实现泛型编程的效果。使得⼀个函数来处理多种类型的数据。
四.const修饰指针
- const用来修饰变量: 如下图:
因为加上const 的 m 就是具有常属性的变量(常属性:不能被修改的属性)虽然 m 不能被修改,但是本质上还是变量,不是常量。可称 m 为常变量
注:但在C++ 中,const修饰的变量为常m量
那 m 怎么才能改呢?具体如下:
- const修饰指针变量:如下图:
上述总结: 当const 修饰指针变量放在*左边时,const限制的是:指针变量指向的内容。指针变量指向的内容不能通过指针来修改,但是可以修改指针变量本身的值(其实修改的是指针变量的指向)
上述总结: 当const 修饰指针变量放在*右边时,const限制的是:指针变量本身。指针变量不能再指向其他变量,但是可以通过指针变量,修改指针变量指向的内容
上述总结: 当const 修饰指针变量放在*左边和右边时,const限制的是:指针变量本身 和 指针变量指向的内容。
五.指针运算
指针的基本运算有三种,分别是:
- 指针 +- 整数
- 指针 - 指针
- 指针的关系运算
指针 +- 整数:
例如:指针 +1 如下:
由以上同理可推出:
type * p
p + 1 -- > 跳过1 * sizeof(type)
p + n-- > 跳过n * sizeof(type)
实践:打印数组中的每个元素。 如下图:
由上述总结得:指针1 +- 整数 == 指针2
指针 - 指针:
注:计算的前提条件:一定是两个指针指向同一块空间!
由上述代码得出:指针 - 指针 == 两个指针之间的绝对值的元素个数
实践: 求字符串的长度。如下图:
上述代码 return str - start; 就是指针 - 指针 得出并返回字符串的长度。
指针的关系运算:就是指针和指针比较大小 如下图:
上述代码通过指针和指针相互比较打印出数组的每一位。
六. 野指针
概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)
野指针的成因:
- 指针未初始化:
- 指针越界访问:
- 指针指向的空间释放:
如何规避野指针:
- 指针要初始化:
- 如果明确知道指针指向哪⾥,就直接初始化一个明确的值。
- 如果不知道指针应该指向哪里,可以给指针赋值NULL。(NULL :是C语⾔中定义的⼀个标识符常量,值是0,0也是地址,这个地址是⽆法使⽤的,读写该地址会报错)
举例:
- 小心指针越界:
⼀个程序向内存申请了哪些空间,通过指针也就只能访问哪些空间,不能超出范围访问,超出了就是越界访问。
- 指针变量不再使用时,及时置NULL,指针使⽤之前检查有效性:
当指针变量指向⼀块区域时,我们可以通过指针访问该区域,后期不再使⽤这个指针访问空间的
时候,我们可以把该指针置为NULL。因为约定俗成的⼀个规则就是:只要是NULL指针就不去访问,同时使⽤指针之前可以判断指针是否为NULL。
我们可以把野指针想象成野狗,野狗放任不管是⾮常危险的,所以我们可以找⼀棵树把野狗拴起来,就相对安全了,给指针变量及时赋值为NULL,其实就类似把野狗栓前来,就是把野指针暂时管理起来。不过野狗即使拴起来我们也要绕着⾛,不能去挑逗野狗,有点危险;对于指针也是,在使⽤之前,我们也要判断是否为NULL,看看是不是被拴起来起来的野狗,如果是,就不能直接使⽤,如果不是,我们再去使⽤。
- 避免返回局部变量的地址。
七.assert断言
assert.h 头⽂件定义了宏 assert() ,用于在运行时确保程序符合指定条件,如果不符合,就报错终止运行。 这个宏常常被称为“断言”
如下:
assert(p != NULL);
上述代码检测:p 是否等于 NULL 。如果确实不等于 NULL ,程序继续运⾏,否则就会终⽌运⾏,并且给出报错信息提示。
assert() 宏接受⼀个表达式作为参数。如果该表达式为真(返回值非零), assert() 不会产⽣
任何作⽤,程序继续运⾏。如果该表达式为假(返回值为零), assert() 就会报错,在标准错误
流 stderr 中写⼊⼀条错误信息,显⽰没有通过的表达式,以及包含这个表达式的⽂件名和⾏号。具体如下:
如果该表达式为假时:
如果该表达式为真时:
注意:assert 不仅可以断言指针还可以断言其他。只要表达式成立就行!
assert() 的使用的好处:
- 出现错误时,能⾃动标识⽂件和出问题的⾏号
- 无需更改代码就能开启或关闭 assert() 的机制(需定义⼀个宏 NDEBUG 。然后重新编译程序,编译器就会禁用文件中所有的 assert() 语句。如果程序又出现问题,可以移除这条 #define NDBUG 指令(或者把它注释掉),再次编译,这样就重启⽤了 assert() 语句。)
举例:
#define NDEBUG 未注释时:
从上述代码可看出:assert()机制未起作用(为关闭状态)
#define NDEBUG 注释时:
从上述代码可看出:assert()机制起作用了(为开启状态)
assert() 的缺点是:因为引入了额外的检查,增加了程序的运行时间。
注:⼀般我们可以在 Debug版本中使⽤#define NDEBUG,在 Release 版本中选择禁⽤ assert 就行,因为在 VS 这样的集成开发环境中,Release 版本中直接就是优化掉了。这样在debug版本写有利于程序员排查问题,在 Release 版本不影响⽤⼾使⽤时程序的效率。
八.指针的使用和传址调用
- strlen(求字符串长度)的模拟实现:
上述代码,参数 str 接收⼀个字符串的起始地址,然后开始统计字符串中 \0 之前的字符个数,最终返回长度。 如果要模拟实现只要从起始地址开始向后逐个字符的遍历,只要不是 \0 字符,计数器就+1,这样直到 \0 就停止了。
- 传值调用和传址调用:
学习指针的目的是使用指针解决问题,那什么问题,非指针不可呢?
例如:写⼀个函数,实现交换两个整型变量的值 :
错误示范:
观察上述代码我们发现:两值并没产⽣交换,这是为什么呢?
看上图,我们通过调试发现:在main函数内部,创建了a和b,a的地址是0x00cffdd0,b的地址是0x00cffdc4,在调⽤ Swap函数时,将a和b传递给了Swap函数,在Swap函数内部创建了形参x和y接收a和b的值,但是 x的地址是0x00cffcec,y的地址是0x00cffcf0,x和y确实接收到了a和b的值,但 x 的地址和 a 的地址不⼀样,y 的地址和 b 的地址不⼀样,相当于x和y是独立的空间。那么在Swap函数内部交换 x 和 y 的值, 自然不会影响 a 和 b,当Swap函数调⽤结束后回到main函数,a和b的没法交换。Swap函数在使用的时候,是把变量本⾝直接传递给了函数,这种调⽤函数的⽅式就叫传值调用。所以Swap是失败的了……
得出结论:实参传递给形参的时候,形参会单独创建⼀份临时空间来接收实参,对形参的修改不影响实参。
正确示范:
上述代码就成功的把两整数 a 和 b 交换了。这⾥调⽤Swap函数的时候是将变量的地址传递给了函数,这种函数调用方式叫:传址调⽤。
总结:传址调⽤,可以让函数和主调函数之间建⽴真正的联系,在函数内部可以修改主调函数中的变量。所以未来在调用函数时,函数中需要修改主调函数中的变量的值,就可以采⽤传址调⽤。如果函数中只需要主调函数中的变量值来实现计算,就可以采用传值调用。
以上就是个人的理解。由于本人水平有限,如有不足之处,恳请各位老师指出。谢谢!
相关文章:

C语言——从头开始——深入理解指针(1)
一.内存和地址 我们知道计算上CPU(中央处理器)在处理数据的时候,是通过地址总线把需要的数据从内存中读取的,后通过数据总线把处理后的数据放回内存中。如下图所示: 计算机把内存划分为⼀个个的内存单元,每…...

微信小程序-绑定数据并在后台获取它
如图 遍历列表的过程中需要绑定数据,点击时候需要绑定数据 这里是源代码 <block wx:for"{{productList}}" wx:key"productId"><view class"product-item" bindtap"handleProductClick" data-product-id"{{i…...

【删除数组用delete和Vue.delete有什么区别】
删除数组用delete和Vue.delete有什么区别? 在 JavaScript 中,delete 和 Vue.js 中的 Vue.delete 是两个完全不同的概念,它们在删除数组元素时的作用和效果也有所不同。 JavaScript 中的 delete 关键字: 在原生 JavaScript 中&a…...

【QT+QGIS跨平台编译】之四十二:【QWT+Qt跨平台编译】(一套代码、一套框架,跨平台编译)
文章目录 一、QWT介绍二、QWT下载三、文件分析四、pro文件五、编译实践5.1 Windows下编译4.2 Linux下编译5.3 MacOS下编译一、QWT介绍 QWT是一个基于Qt框架的开源C++库,用于创建交互式的图形用户界面。它提供了丰富的绘图和交互功能,可以用于快速开发图形化应用程序。 QWT包…...

yum方式快速安装mysql
问题描述 使用yum的方式简单安装了一下mysql,对过程进行简单记录。 步骤 ①安装wget和vim sudo yum -y install wget vim②下载mysql的rpm包 sudo wget https://dev.mysql.com/get/mysql80-community-release-el7-3.noarch.rpm③升级和更新rpm包 sudo rpm -Uv…...

基于Java的家政预约管理平台
功能介绍 平台采用B/S结构,后端采用主流的Springboot框架进行开发,前端采用主流的Vue.js进行开发。 整个平台包括前台和后台两个部分。 前台功能包括:首页、家政详情、家政入驻、用户中心模块。后台功能包括:家政管理、分类管理…...

C语言前世今生
C语言前世今生 C语言的发展历史 C语言于1972年11月问世,1978年美国电话电报公司(AT&T)贝尔实验室正式发布C语言,1983年由美国国家标准局(American National Standards Institute,简称ANSI)…...

android aidl进程间通信封装通用实现-用法说明
接上一篇:android aidl进程间通信封装通用实现-CSDN博客 该aar包的使用还是比较方便的 一先看客户端 1 初始化 JsonProtocolManager.getInstance().init(mContext, "com.autoaidl.jsonprotocol"); //客户端监听事件实现 JsonProtocolManager.getInsta…...

【Java中23种设计模式-单例模式2--懒汉式线程不安全】
加油,新时代打工人! 今天,重新回顾一下设计模式,我们一起变强,变秃。哈哈。 23种设计模式定义介绍 Java中23种设计模式-单例模式 package mode;/*** author wenhao* date 2024/02/19 09:16* description 单例模式--懒…...

【后端高频面试题--Linux篇】
🚀 作者 :“码上有前” 🚀 文章简介 :后端高频面试题 🚀 欢迎小伙伴们 点赞👍、收藏⭐、留言💬 后端高频面试题--Linux篇 往期精彩内容Windows和Linux的区别?Unix和Linux有什么区别…...

网络原理HTTP/HTTPS(2)
文章目录 HTTP响应状态码200 OK3xx 表示重定向4xx5xx状态码小结 HTTPSHTTPS的加密对称加密非对称加密 HTTP响应状态码 状态码表⽰访问⼀个⻚⾯的结果.(是访问成功,还是失败,还是其他的⼀些情况…).以下为常见的状态码. 200 OK 这是⼀个最常⻅的状态码,表⽰访问成功 2xx都表示…...

【Java中23种设计模式-单例模式2--懒汉式2线程安全】
加油,新时代打工人! 简单粗暴,学习Java设计模式。 23种设计模式定义介绍 Java中23种设计模式-单例模式 Java中23种设计模式-单例模式2–懒汉式线程不安全 package mode;/*** author wenhao* date 2024/02/19 09:38* description 单例模式…...

由LeetCode541引发的java数组和字符串的转换问题
起因是今天在刷下面这个力扣题时的一个报错 541. 反转字符串 II - 力扣(LeetCode) 这个题目本身是比较简单的,所以就不讲具体思路了。问题出在最后方法的返回值处,要将字符数组转化为字符串,第一次写的时候也没思考直…...

HTTP 头部- Origin Referer
Origin & Referer Origin Header 示例 Origin 请求头部是一个 HTTP 头部,它提供了发起请求的网页的源(协议、域名和端口)信息。它通常在进行跨域资源共享(CORS)请求时使用,以便服务器可以决定是否接受…...

Python 实现Excel 文件合并
Excel 文件合并方法较多,前面文章有通过Uipath RPA 对文件进行合并,也可以通过Python或VBA写脚本合并。 通常写脚本维护性更加简洁,本文提供Python 脚本对Excel 文件进行合并,参考Uipath 调用Python 文章,Uipath 调用Python 脚本程序详解-CSDN博客 便能快速实现。代码如…...

ECMAScript 6+ 新特性 ( 一 )
2.1.let关键字 为了解决之前版本中 var 关键字存在存在着越域, 重复声明等多种问题, 在 ES6 以后推出 let 这个新的关键字用来定义变量 //声明变量 let a; let b,c,d; let e 100; let f 123, g hello javascript, h [];let 关键字用来声明变量,使用 let 声明的…...

动态DP入门线性动态DP
动态DP入门&线性动态DP 前言核心思想例1例22024牛客寒假4K2022牛客寒假2J结论 前言 OI-WiKi上有一个动态DP讲解,直接讲到了树型DP领域,同时需要树链剖分,门槛有点高。本文针对线性DP做一个动态DP的讲解。 首先当然要懂得一定的DP的相关…...

基于python+django+vue.js开发的停车管理系统
功能介绍 平台采用B/S结构,后端采用主流的Python语言进行开发,前端采用主流的Vue.js进行开发。 功能包括:车位管理、会员管理、停车场管理、违规管理、用户管理、日志管理、系统信息模块。 源码地址 https://github.com/geeeeeeeek/pytho…...

网站管理新利器:免费在线生成 robots.txt 文件!
🤖 探索网站管理新利器:免费在线生成 robots.txt 文件! 你是否曾为搜索引擎爬虫而烦恼?现在,我们推出全新的在线 robots.txt 文件生成工具,让你轻松管理网站爬虫访问权限,提升网站的可搜索性和…...

【Java程序员面试专栏 Java领域】Java虚拟机 核心面试指引
关于Java 虚拟机部分的核心知识进行一网打尽,主要包括Java虚拟机的内存分区,执行流程等,通过一篇文章串联面试重点,并且帮助加强日常基础知识的理解,全局思维导图如下所示 JVM 程序执行流程 包括Java程序的完整执行流程,以及Javac编译,JIT即时编译 Java程序的完整执…...

洛谷C++简单题小练习day15—计算阶乘小程序(不用循环)
day15--计算阶乘小程序--2.19 习题概述 题目描述 求 n!,也就是 123⋯n。 挑战:尝试不使用循环语句(for、while)完成这个任务。 输入格式 第一行输入一个正整数 n。 输出格式 输出一个正整数,表示 n! 代码部分 …...

Vue报错,xxx is defined #变量未定义
vue.js:5129 [Vue warn]: Error in v-on handler: "ReferenceError: count is not defined" 浏览器将这个变量 当做全局变量了,事实上它只是实例中的变量 加上this指定,是vue实例中的变量...

Idea启动Gradle报错: Please, re-import the Gradle project and try again
Idea启动Gradle报错:Warning:Unable to make the module: reading, related gradle configuration was not found. Please, re-import the Gradle project and try again. 解决办法: 开启步骤:View -> Tool Windows -> Gradle 点击refe…...

Python函数(一)
目录 一、定义函数 (一)向函数传递信息 (二)实参和形参 二、传递实参 (一)位置实参 (二)关键字实参 (三)默认值 (四)等效的函…...

Excel表的内容批量生成个人加水印的Word文档
Excel表的内容批量生成个人加水印的Word文档 以下代码可以直接复制到docm文件里使用 Sub 宏1()Dim MyDialog As FileDialogDim GetStr As String, Adoc As StringDim PsDoc As DocumentApplication.ScreenUpdating FalseSet MyDialog Application.FileDialog(msoFileDialogF…...

微服务设计:Spring Cloud API 网关概述
Spring Cloud API 网关是指一个位于微服务架构中的代理服务器,它负责将外部请求路由到内部微服务。API 网关可以提供多种功能,包括: 路由: 将请求路由到特定的微服务。负载均衡: 将请求分散到多个微服务实例上。安全: 身份验证、授权和安全策…...

stm32学习笔记-STLINK使用
stm32学习笔记-STLINK使用 使用ST-LINK调试程序进度表格 使用ST-LINK调试程序 说明 组成 总结 记录使用STLINK进行项目的烧写和调试,旨在高效的进行代码调试学习工具包括笔记本、keil5MDK、stm32f030c8t6电表主机、STLINK V2、导线、电表代码总的来说࿰…...

Linux CentOS stream 9 firewalld
随着互联网行业快速发展,服务器成为用户部署网络业务重要的网络工具,但随之而来的就是更密集的网络攻击,这给网站带来了很大的阻碍。防火墙作为保障网络安全的主要设备,可以很好的抵御网络攻击。 防火墙基本上使用硬件和软件两种…...

VLM多模态图像识别小模型UForm
参考:https://github.com/unum-cloud/uform https://huggingface.co/unum-cloud/uform-gen2-qwen-500m https://baijiahao.baidu.com/s?id=1787054120353641459&wfr=spider&for=pc demo:https://huggingface.co/spaces/unum-cloud/uform-gen2-qwen-500m-demo UF…...

我的NPI项目之设备系统启动(七) -- 高通OS启动阶段和镜像分区简析
每当有新的平台起来的时候,大概率会伴随着新系统的发布,无论是高通的还是Google Andorid的。在做平台Bringup阶段总会遇到各种各样的专业术语。例如,总是会听到有人说PBL,SBL,XBL,UFEI,Bootload…...