当前位置: 首页 > news >正文

C++指针详解

旧文更新:两三年的旧文了,一直放在电脑里,现在直接传上CSDN

一、指针的概念

1.1 指针

程序运行时每个变量都会有一块内存空间,变量的值就存放在这块空间中。程序可以通过变量名直接访问这块空间内的数据,这种访问方式称之为直接访问。

而内存中每一个字节都会有一个编号,称之为内存地址。我们可以通过直接访问内存地址从而找到变量。在C++中,有一种指针变量专门用于存储内存地址,我们通过访问指针变量,可以得到对应变量的内存地址,从而实现对该变量的访问。这就是间接访问。

指针存储的是一个内存地址,他的一个重要用途就是通过指针间接访问所指向的地址中的内容。指针定义如下:
类型名 *指针变量名;
比如定义两个指针p1和p2:
int *p1, *p2;

1.2 指针的基本操作

1.指针变量的赋值
指针变量一般用于保存同一个程序中某个变量的地址,以便日后使用指针变量进行间接访问。因此赋值有两种办法:一是将本程序的某一变量地址赋值给指针变量,另一种是将指针变量赋值给另一个指针变量。

让指针变量只想某一变量,就是将一个变量的地址存入指针变量,但是程序员并不知道变量在内存中的地址,为此C++提供了一个取地址运算符&,用于获取指定变量的地址,例如:

int *p, x;
p=&x;

这样就成了将x变量的地址放入到指针p中,上述语句等价于int x, *p=&x

和普通变量一样,如果没有给新定义的指针变量赋值,C++会给它随机赋以初址,这会导致内存泄漏,因此为了避免误操作,不要引用没有被赋值的指针,并且某个指针被创建了但暂且不用,应该给它赋以NULL值。

需要注意的是,不同变量类型的指针之间不能相互赋值,比如int x;double *p =x;这种操作是不允许的,因为指针是使用变量类型大小来判断边界的。在内存中int是4字节大小的,而double是8字节大小的。那么double指针会以其存储的地址为起点,读取往后的8个字节的内容,很明显,这超过了int型的大小限制。

2.指针变量的访问
C++中定义了一个取指针指向值的运算符*,该运算符是一个一元运算符,他的运算对象是一个指针,运算符会根据指针的类型返回其指向的值。这在指针定义之后使用。
也就是说指针在定义的时候需要在变量名前加*,但是在后续使用指针变量时候是不需要加*的,指针p在被定义后再使用*p的话会直接得出指针所指的变量的值,下列是一个简单的实例

int a=2, *p=&a;
cout<<*p<<" "<<p<<" "<<a;

我们可以看到输出结果是:2 0x16d81b868 2。也就是在完成定义后,p表示内存地址,*p表示该内存地址存入的值。

3.统配指针类型void
指针的基本类型可以为void,这只说明了这个变量中存放的是内存地址,但是并未说明存放何种类型的地址,因此任何类型的指针都可以和void类型指针相互赋值。

4.指针和常量限定符const
使用const修饰指针有如下三种用法:
1.指向常量的指针(常量指针):是一个执行内容是常量的指针变量,指针可以更改指向的对象,但是不能改变指向对象的值。

const int *p = &x;

2.指针常量:指针本身是一种常量,无法改变指针的值,但是可以改变指向的变量的值

int *const p=&x;

3.指向常量的指针常量:指针本身不可变,所指向的值也不能变

const int *const p = &x;

二、指针和数组

2.1 指针运算

指针保存的是一个内存地址,本质上是一个整数,因此对整数进行算术运算也是十分理算当然得。但是对于指针执行乘除运算是没有意义的,因为这会使得指针指向不可知的领域,C++中只考虑了指针的加减运算。对指针执行+1,那么他会增加一个基本类型的长度,比如对一个int *p=1000执行p++,那么p=1004,因为整形变量的大小为4。

2.2 用指针访问数组

在C++中,一个数组的数组名保存的是数组的起始地址,也就是说,实际上数组名就是一个指针,不过它是一个指针常量。不过数组在声明时除了将数组首地址赋值给了数组名外,还进行了空间申请的行为。我们完全可以将数组名赋值给一个指针,那么该指针就具备了数组名的行为

三、动态内存分配

3.1 动态变量

有时候在C++中我们需要动态地新增变量,或者说动态指定数组的大小,这就需要使用到动态变量机制了。动态变量指的是在写成是时无法确定它们的存在,只有程序运行起来,随着程序的运行可以根据程序的需求动态产生和消亡的变量。由于动态变量不可以在程序中定义,因此也没法给他们取名字,因此动态变量的访问需要通过指向动态变量的指针来实现间接访问

3.2 动态变量的创建

C++的动态变量的创建使用的运算符是new,该运算符可以创建一个简单变量或者一个数组,格式如下:
new 类型名;
这个操作在内存中称之为堆的区域申请一块能够存放相应数据类型的数据空间,完成操作后会返回该空间的首地址。比如要动态产生一个int型变量,并且将20赋值给该变量,则操作如下:

int *p;
p = new int;
*p = 20;

需要注意,new操作是需要指定类型的,因此它的结果只能赋值给同类指针。
用new操作可以创建一个一维数组,格式如下:
new 类型名[元素个数];
动态数组和普通数组最大的区别在于,数组的大小不需要静态指明一个常量,可以在运行时赋予,因此更加灵活。比如:

p = new int[2*n];

3.3 动态变量的消亡

在C++运行期间,动态变量不会消亡,因此要回收动态变量的空间就必须显式地使之消亡。要回收某个动态变量,可以使用delete操作。操作如下:
delete 指针变量
该操作会回收指针指向的空间,例子如下:

int *p = new int(10);
delete p;

如果需要回收的是一个动态数组,可以使用delete [] 指针变量
一旦释放了这些内存区域,堆管理器会重新回收这些区域,虽然指针还会指向这个区域,但是已经不能再访问和使用这些区域

在动态变量的使用中,最常见的问题是内存泄漏,也就是用动态变量机制申请了一个动态变量,但是在不需要使用这个动态变量的时候没有delete,或者在变量没有delete之前,让该动态变量的指针指向另外一个动态变量,此时你就彻底弄丢这个动态变量了!堆管理器以为你仍在使用它,但是你自己都找不到它在哪了。为了避免上述变量,应该及时使用delete清除不需要使用的动态变量。

四、指针和函数

4.1 指针作为形式参数

函数的参数可以传入指针变量,作用是将一个变量的地址传入到一个函数中,指针传递可以降低参数传递的开销,并且让主函数和被调用函数共享一块内存空间,指针作为形式参数是一种”址传递“,在函数中对指针所指向的变量进行操作,也会改变该变量的值,此时在主函数中读取该变量也是被改变的。而一般参数的”值传递“,传递的形式参数的作用域只会是在被调用函数内部,对值传递的参数进行修改只会在被调函数中起改变,而不会影响到主函数。

但是数组名是一种特殊的情况,当数组名作为函数的参数时,形式参数和实际参数是共享了同一块内存空间,因此函数内对数组形式参数的任何修改都是对实际参数数组的修改。

字符串是作为字符数组存储的,因此传递进去的也是指针,但是字符串会以’\0’结尾,所以可以不传递字符数组的长度

4.2 返回指针的函数

函数返回值可以是一个指针,表示函数的返回值是一个指针只需要在函数名之前加一个*号,此时返回的值必须是一个指针变量。比如:
在这里插入图片描述

五、引用类型

5.1 引用类型

指针提供了通过一个变量间接访问另一个变量的能力。为了获得指针的效果,又要避免指针的问题,C++提供了引用类型这种新的类型。

1.引用的声明和使用
引用就是给变量取一个别名,使得一块内存空间可以通过几个变量名在访问。声明引用类型的变量需要在变量名前加上符号&,并且必须指定初值。比如:

int i;
int &j = i;

其中第二个语句中定义了变量j是变量i的别名,当编译器遇到这个语句的时候,他并不会为变量j分配空间,而是只是将i和j的地址关联起来,i和j用的是同一内存单元,通过j可以访问i的空间。引用实际上是一种隐式指针,每次使用引用变量可以不用书写运算符"*",因此简化了程序的书写。

定义引用类型变量时,必须在变量名前加上符号&用于区别和普通变量的区别。并且定义时就必须对其进行初始化并赋值。

5.2 引用传递

1.引用传递参数
引用的主要目的是将引用作为参数的函数,最经典的是swap函数,如果swap函数使用指针书写,则如下:

void swap(int* a, int* b){int c;c = *a;*a = *b;*b = c;
}

很显然,如果使用指针传递,那么需要在每一个运算符前都加上指针运算符*,但是使用引用传递只需要像如下一样书写:

void swap(int& a, int& b){int c;c = a;a = b;b = c;
}

可以看到,使用引用符号后,不需要书写这么多*号了。在使用引用类型参数的时候需要注意,调用时对应的实际参数必须为左值表达式

5.3 返回引用的函数

函数的返回值也可以是一个引用,代表返回函数内某个变量的引用。

相关文章:

C++指针详解

旧文更新&#xff1a;两三年的旧文了&#xff0c;一直放在电脑里&#xff0c;现在直接传上CSDN 一、指针的概念 1.1 指针 程序运行时每个变量都会有一块内存空间&#xff0c;变量的值就存放在这块空间中。程序可以通过变量名直接访问这块空间内的数据&#xff0c;这种访问方…...

tauri+vite+vue3开发环境下创建、启动运行和打包发布

目录 1.创建项目 2.安装依赖 3.启动项目 4.打包生成windows安装包 5.安装打包生成的安装包 1.创建项目 运行下面命令创建一个tauri项目 pnpm create tauri-app 我创建该项目时的node版本为16.15.0 兼容性注意 Vite 需要 Node.js 版本 14.18&#xff0c;16。然而&#x…...

安卓进阶系列-系统基础

文章目录计算机结构冯诺依曼结构哈弗结构冯诺依曼结构与哈弗结构对比安卓采用的架构安卓操作系统进程间通讯&#xff08;IPC&#xff09;内存共享linux内存共享安卓内存共享管道Unix Domain Socket同步常见同步机制信号量Mutex管程安卓同步机制安卓中的Mutex安卓中的ConditionB…...

10 Wifi网络的封装

概述 Wifi有多种工作模式,比如:STA模式、AccessPoint模式、Monitor模式、Ad-hoc模式、Mesh模式等。但在IPC设备上,主要使用STA和AccessPoint这两种模式。下面分别进行介绍。 STA模式:任何一种无线网卡都可以运行在此模式,这种模式也是无线网卡的默认模式。在此模式下,无线…...

手把手的教你安装PyCharm --Pycharm安装详细教程(一)(非常详细,非常实用)

简介 Jetbrains家族和Pycharm版本划分&#xff1a; pycharm是Jetbrains家族中的一个明星产品&#xff0c;Jetbrains开发了许多好用的编辑器&#xff0c;包括Java编辑器&#xff08;IntelliJ IDEA&#xff09;、JavaScript编辑器&#xff08;WebStorm&#xff09;、PHP编辑器&…...

开发板与ubantu文件传送

接下来的所以实验都通过下面这种方式发送APP文件到开发板运行 目录 1、在ubantu配置 ①在虚拟机上添加一个桥接模式的虚拟网卡 ②设定网卡 ③在网卡上配置静态地址 2、开发板设置 ①查看网卡 ②配置网卡静态ip 3、 测试 ①ping ②文件传送 传送报错情况 配置环境&#…...

如何成为一名优秀的网络安全工程师?

前言 这是我的建议如何成为网络安全工程师&#xff0c;你应该按照下面顺序学习。 简要说明 第一件事你应该学习如何编程&#xff0c;我建议首先学python&#xff0c;然后是java。 &#xff08;非必须&#xff09;接下来学习一些算法和数据结构是很有帮助的&#xff0c;它将…...

面试问题之高并发内存池项目

项目部分 1.这个项目是什么? 高并发内存池的原型是谷歌一个开源项目&#xff0c;tcmalloc&#xff0c;而这个项目&#xff0c;就是tcmalloc中最核心的框架和部分拿出来进行模拟。他的作用就是在去代替原型的内存分配函数malloc和free。这个项目涉及的技术有&#xff0c;c&…...

如果阿里巴巴给蒋凡“百亿补贴”

出品 | 何玺 排版 | 叶媛 2021底&#xff0c;阿里内部进行组织架构大调整&#xff0c;任命蒋凡为阿里海外商业负责人&#xff0c;分管全球速卖通和国际贸易&#xff08;ICBU&#xff09;两个海外业务&#xff0c;以及Lazada等面向海外市场的多家子公司。 一年时间过去&#x…...

Linux版本现状

Linux的发行版本可以大体分为两类&#xff0c;一类是商业公司维护的发行版本&#xff0c;一类是社区组织维护的发行版本&#xff0c;前者以著名的Red Hat&#xff08;RHEL红帽&#xff09;为代表&#xff0c;后者以Debian为代表。Red HatRedhat&#xff0c;应该称为Redhat系列&…...

Winform中实现保存配置到文件/项目启动时从文件中读取配置(序列化与反序列化对象)

场景 Winform中实现序列化指定类型的对象到指定的Xml文件和从指定的Xml文件中反序列化指定类型的对象&#xff1a; Winform中实现序列化指定类型的对象到指定的Xml文件和从指定的Xml文件中反序列化指定类型的对象_winform xml序列化_霸道流氓气质的博客-CSDN博客 上面讲的序…...

基于python的超市历年数据可视化分析

人生苦短 我用python Python其他实用资料:点击此处跳转文末名片获取 数据可视化分析目录人生苦短 我用python一、数据描述1、数据概览二、数据预处理0、导入包和数据1、列名重命名2、提取数据中时间&#xff0c;方便后续分析绘图三、数据可视化1、美国各个地区销售额的分布&…...

GPT-4技术报告

摘要 链接&#xff1a;https://cdn.openai.com/papers/gpt-4.pdf 我们汇报了GPT-4的发展&#xff0c;这是一个大规模的多模态模型&#xff0c;可以接受图像和文本输入并产生文本输出。虽然在许多现实场景中&#xff0c;GPT-4的能力不如人类&#xff0c;但它在各种专业和学术基…...

前端性能优化

总结 使用打包工具对代码进行打包压缩&#xff1b;引入css时采用link标签&#xff0c;并放入头部&#xff0c;使其与文档一起加载&#xff0c;减少页面卡顿时间&#xff1b;尽量减少dom结构的重排和重绘&#xff1b;使用css雪碧图&#xff0c;减少网络请求&#xff1b;对不同分…...

尚医通-(三十三)就诊人管理功能实现

目录&#xff1a; &#xff08;1&#xff09;前台用户系统-就诊人管理-需求说明 &#xff08;2&#xff09;就诊人管理-接口开发-列表接口 &#xff08;3&#xff09;就诊人管理-接口开发-其他接口 &#xff08;4&#xff09;前台用户系统-就诊人管理-前端整合 &#xff0…...

《Spring Boot 趣味实战课》读书笔记(二)

牛刀小试——五分钟入门 Spring Boot 万物皆可 Hello World 创建一个 Web 工程 填写项目信息 选择依赖 从 IDEA 打开下载好的 Spring Boot 工程&#xff1a; 完成核心代码 创建 HelloController 类并编写 hello 方法 创建一个 HelloController 类&#xff0c;或者选择 Fi…...

Spring Cloud -- GateWay

为什么需要网关在微服务架构中&#xff0c;一个系统会被拆分为很多个微服务。那么作为客户端要如何去调用这么多的微服务呢&#xff1f;如果没有网关的存在&#xff0c;我们只能在客户端记录每个微服务的地址&#xff0c;然后分别去调用。这样的话会产生很多问题&#xff0c;例…...

【C语言】memcpy , memset等内存操作函数使用方法与注意事项

这个章节&#xff0c;我们探讨C语言内存操作函数。 重点介绍处理内存操作函数使用和注意事项 和内存函数如何模拟实现。 内存函数所需头文件 #include<string.h> 文章目录memcpymemcpy 函数模拟实现memmovememmove 函数模拟实现memcmpmemcmp 函数模拟实现memsetmemset 函…...

尚融宝04-mybatis-plus插件和条件构造器

目录 一、分页插件 1、添加配置类 2、添加分页插件 3、测试分页 二、XML自定义分页 1、UserMapper中定义接口方法 2、定义XML 3、测试 三、乐观锁 1、场景 2、乐观锁方案 3、乐观锁实现流程 4、优化流程 四、wapper介绍 1、Wrapper家族 2、创建测试类 五、Qu…...

面试重难点问题(C++)

持续更新&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 网络部分 1.问&#xff0c;四次挥手的过程&#xff0c;和双方状态变化&#xff1f; 挥手这前&#xff0c;两边都是established状态&#xff0c;客户端发起断开请求&#xff0c;向服务器发送fin请求&…...

C++.OpenGL (10/64)基础光照(Basic Lighting)

基础光照(Basic Lighting) 冯氏光照模型(Phong Lighting Model) #mermaid-svg-GLdskXwWINxNGHso {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-GLdskXwWINxNGHso .error-icon{fill:#552222;}#mermaid-svg-GLd…...

鸿蒙DevEco Studio HarmonyOS 5跑酷小游戏实现指南

1. 项目概述 本跑酷小游戏基于鸿蒙HarmonyOS 5开发&#xff0c;使用DevEco Studio作为开发工具&#xff0c;采用Java语言实现&#xff0c;包含角色控制、障碍物生成和分数计算系统。 2. 项目结构 /src/main/java/com/example/runner/├── MainAbilitySlice.java // 主界…...

MySQL账号权限管理指南:安全创建账户与精细授权技巧

在MySQL数据库管理中&#xff0c;合理创建用户账号并分配精确权限是保障数据安全的核心环节。直接使用root账号进行所有操作不仅危险且难以审计操作行为。今天我们来全面解析MySQL账号创建与权限分配的专业方法。 一、为何需要创建独立账号&#xff1f; 最小权限原则&#xf…...

Yolov8 目标检测蒸馏学习记录

yolov8系列模型蒸馏基本流程&#xff0c;代码下载&#xff1a;这里本人提交了一个demo:djdll/Yolov8_Distillation: Yolov8轻量化_蒸馏代码实现 在轻量化模型设计中&#xff0c;**知识蒸馏&#xff08;Knowledge Distillation&#xff09;**被广泛应用&#xff0c;作为提升模型…...

基于 TAPD 进行项目管理

起因 自己写了个小工具&#xff0c;仓库用的Github。之前在用markdown进行需求管理&#xff0c;现在随着功能的增加&#xff0c;感觉有点难以管理了&#xff0c;所以用TAPD这个工具进行需求、Bug管理。 操作流程 注册 TAPD&#xff0c;需要提供一个企业名新建一个项目&#…...

代码规范和架构【立芯理论一】(2025.06.08)

1、代码规范的目标 代码简洁精炼、美观&#xff0c;可持续性好高效率高复用&#xff0c;可移植性好高内聚&#xff0c;低耦合没有冗余规范性&#xff0c;代码有规可循&#xff0c;可以看出自己当时的思考过程特殊排版&#xff0c;特殊语法&#xff0c;特殊指令&#xff0c;必须…...

渗透实战PortSwigger靶场:lab13存储型DOM XSS详解

进来是需要留言的&#xff0c;先用做简单的 html 标签测试 发现面的</h1>不见了 数据包中找到了一个loadCommentsWithVulnerableEscapeHtml.js 他是把用户输入的<>进行 html 编码&#xff0c;输入的<>当成字符串处理回显到页面中&#xff0c;看来只是把用户输…...

Kubernetes 节点自动伸缩(Cluster Autoscaler)原理与实践

在 Kubernetes 集群中&#xff0c;如何在保障应用高可用的同时有效地管理资源&#xff0c;一直是运维人员和开发者关注的重点。随着微服务架构的普及&#xff0c;集群内各个服务的负载波动日趋明显&#xff0c;传统的手动扩缩容方式已无法满足实时性和弹性需求。 Cluster Auto…...

英国云服务器上安装宝塔面板(BT Panel)

在英国云服务器上安装宝塔面板&#xff08;BT Panel&#xff09; 是完全可行的&#xff0c;尤其适合需要远程管理Linux服务器、快速部署网站、数据库、FTP、SSL证书等服务的用户。宝塔面板以其可视化操作界面和强大的功能广受国内用户欢迎&#xff0c;虽然官方主要面向中国大陆…...

联邦学习带宽资源分配

带宽资源分配是指在网络中如何合理分配有限的带宽资源&#xff0c;以满足各个通信任务和用户的需求&#xff0c;尤其是在多用户共享带宽的情况下&#xff0c;如何确保各个设备或用户的通信需求得到高效且公平的满足。带宽是网络中的一个重要资源&#xff0c;通常指的是单位时间…...