[C++基础学习-06]----C++指针详解

前言

        指针是一个存储变量地址的变量,可以用来访问内存中的数据。在C++中,指针是一种非常有用的数据类型,可以帮助我们在程序中对内存进行操作和管理。

正文

01-指针简介    

        指针的基本概念如下:

        声明指针:使用“*”符号来声明指针变量,如int *ptr;表示声明一个指向整型变量的指针。

        给指针赋值:可以使用&运算符来获取变量的地址,并将其赋给指针变量,如int a = 10; int *ptr = &a;表示将整型变量a的地址赋给指针ptr。

        使用指针访问变量:可以使用*运算符来获取指针指向的变量的值,如int value = *ptr;表示获取指针ptr所指向的整型变量的值。

        修改指针指向的变量:可以直接对指针指向的变量进行修改,如*ptr = 20;表示将指针ptr所指向的整型变量的值修改为20。

        指针的使用过程中需要注意一些注意事项:

        尽量避免野指针:未初始化或指向无效地址的指针会导致程序崩溃或产生未定义行为,因此应该始终确保指针指向有效的内存位置。

        避免内存泄漏:使用完指针后应该及时释放内存,否则会导致内存泄漏,造成系统资源浪费。

        指针类型要匹配:指针的类型应该与其所指向的变量类型匹配,否则会导致编译错误或程序运行时出错。

02-指针的定义和使用    

        具体代码和解释如下:

#include<iostream>
using namespace std;int main()
{// 定于一个指针   指针就是一个地址,不是一个值int a = 10;// 1、指针定义的语法:数据类型  * 指针变量名:int *p;p = &a;   //这里直接将a的地址传递给了pcout << "a的地址为:" << &a << endl;cout << "p的地址为:" << p << endl;// 上述只是p得到a的地址,2、若想得到a的内存,通过解引用的方式可以获得内存// 指针前加 * 代表解引用,找到指针指向的内存中的数据*p = 1000; // 这里将a和p同时更改cout << "a = " << a << endl;cout << "p = " << *p << endl;system("pause");return 0;
}

03-指针所占空间    

        具体代码和解释如下:

#include<iostream>
using namespace std;int main()
{// 指针所占内存空间大小int a = 10;int * p = &a;// 指针在32位操作系统中占有4个字节  无论什么数据类型,都是4个字节 X86 32位  X64 64位cout << "sizeof (int *) = " << sizeof(int*) << endl;   //这里写int* 和 p是一样的输出方式  cout << "sizeof (int *) = " << sizeof(float*) << endl;cout << "sizeof (int *) = " << sizeof(double*) << endl;cout << "sizeof (int *) = " << sizeof(char*) << endl;system("pause");return 0;
}

04-空指针所占空间    

        具体代码和解释如下:

#include<iostream>
using namespace std;int main()
{// 空指针// 1、空指针用于给指针变量进行初始化int *p = NULL;   //  NULL代表0// 2、空指针是不允许访问的// 0~255之间的内存是系统占用的,不可以进行访问// *p = 100; system("pause");return 0;
}

05-野指针    

        具体代码和解释如下:

#include<iostream>
using namespace std;int main()
{// 野指针// 在主程序中尽量避免出现int *p = (int*)0x1100;   //非法空间cout << *p << endl;system("pause");return 0;
}

06-const修饰指针    

        具体代码和解释如下:

#include<iostream>
using namespace std;int main()
{// const修饰指针的三种情况/* 1、常量指针2、指针常量3、const既修饰指针,又修饰常量*/int a = 10;int b = 10;const int *p = &a;   /* 1、常量指针此时上述代码中的p 使用解引用 获取p的值 *p 但是*p不能进行更改  *p = 20 报错而 p = &b 这样可以更改,也就是说,p的指向可以更改  这叫常量指针*/int * const p = &a;/* 2、指针常量此时指的是指针常量,因为,const后加变量指的是常量,这时 *p 可以更改  *p = 20 正确但是 p 的指向不能更改,也就是说 p = &b 将报错 */const int * const p = &a;/* 3、const既修饰指针,又修饰常量此时无论指针指向还是指向的值都不可以更改*/system("pause");return 0;
}

07-指针与数组    

        具体代码和解释如下:

#include<iostream>
using namespace std;int main()
{// 利用指针访问数组中的元素int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };cout << "第一个元素为:" << arr[0] << endl;cout << "第一个元素地址为:" << &arr[0] << endl;cout << "第一个元素占有字节为:" << sizeof(arr[0]) << endl;  // 4个字节int *p = arr;   // 因为数组名arr就是首地址for (int i = 0; i < 10;i++){cout <<  *p<< endl;p++;  // p++一次也就是地址向前加4个字节 ,数组中每一个元素占有四个字节,因此p++相当于从一个元素到下一个元素}system("pause");return 0;
}

08-指针与函数    

        具体代码和解释如下:

#include<iostream>
using namespace std;// 实现两个数字的交换
void swap01(int a, int b)
{int temp = a;a = b;b = temp;cout << "a =" << a << endl;cout << "b = " << b << endl;
}void swap02(int *p1, int *p2)
{int temp = *p1;*p1 = *p2;*p2 = temp;}int main()
{// 指针和函数// 1、值传递int a = 10;int b = 20;
//	swap01(a, b);// 2、地址传递swap02(&a, &b);cout << "a =" << a << endl;   // 20cout << "b = " << b << endl;  // 10/* 上述经过地址传递传递之后,a和b实参也发生了改变,具体原因如下1、首先指针一直存储的都是一个地址,比如 p 就是一个地址,但是加上 *p 之后,这就使用了解引用,此时指的是指针p指向的数值,而不是地址。2、上述代码中首先,调用swap02()函数时,使用将a和b的地址传递给p1和p2,此时 p1=a的地址,p2=b的地址。而*p1=10,*p2=203、swap02()函数时进行数值更换的函数,*p1表示对应a的地址指向的a的数值,*p2指的是对应于b的地址指向的b的数值,而该函数,将*p2赋值给了*p1,*p1赋值给了*p2造成a和b的地址发生改变,地址对应的数值也发生改变  传递地址,将间接改变实参的数据*/system("pause");return 0;
}

09-指针数组函数案例    

        具体代码、解释和运行结果如下:

#include <iostream>
using namespace std;// 创建冒泡排序函数void bubbleSort(int *arr, int len)
{for (int i = 0; i < len-1;i++){for (int j = 0; j < len - i - 1;j++){// 如果arr[j]>arr[j+1],则两个数进行交换if (arr[j]>arr[j+1]){int temp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = temp;}}}
}// 创建打印数组函数
void printArray(int *arr, int len)
{for (int i = 0; i < len;i++){cout << arr[i] << endl;}
}int main()
{// 1、首先创建数组int arr[10] = { 4,3,6,9,1,2,10,8,7,5 };// 为了方便在函数中使用,并且数组长度可能回不一致,因此可以创建一个变量,用于存储数组长度int len = sizeof(arr) / sizeof(arr[0]);// 2、创建函数,实现冒泡排序,调用排序函数/*  因为数组名代表的就是数组的首地址,也就是第一个元素的地址,因此,定义函数时,也需要定义指针,而调用函数时由于本身就是地址,没有必要再进行取地址操作*/bubbleSort(arr, len);// 3、打印排序后的数据,可以直接创建一个打印函数printArray(arr, len);system("pause");return 0;}

总结

        指针是C++中非常重要的概念,可以帮助我们更灵活地对内存进行操作,但在使用指针时需要注意避免常见的问题,以确保程序的稳定性和安全性。

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

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

相关文章

【考研数学】武忠祥「基础篇」如何衔接进入强化?

如果基础篇已经做完&#xff0c;并且讲义上的例题也都做完了&#xff0c; 那下一步就是该做题了 这个时候&#xff0c;不能盲目做题&#xff0c;做什么题很重要&#xff01;我当初考研之前&#xff0c;基础也很差&#xff0c;所以考研的时候选了错误的题集&#xff0c;做起来就…

设计网页用什么软件

在设计网页时&#xff0c;可以使用多种软件来完成不同的任务。以下是一些常用的网页设计软件&#xff0c;以及它们的特点和用途。 1. Adobe Photoshop&#xff1a; Adobe Photoshop 是一款功能强大的图像编辑软件。在网页设计中&#xff0c;它常用于创建和编辑网页所需的图像、…

5-在Linux上部署各类软件

1. MySQL 数据库安装部署 1.1 MySQL 5.7 版本在 CentOS 系统安装 注意&#xff1a;安装操作需要 root 权限 MySQL 的安装我们可以通过前面学习的 yum 命令进行。 1.1.1 安装 配置 yum 仓库 # 更新密钥 rpm --import https://repo.mysql.com/RPM-GPG-KEY-mysql-2022# 安装Mysql…

C/C++ BM33 二叉树的镜像

文章目录 前言题目解决方案一1.1 思路阐述1.2 源码 总结 前言 镜像说的好听&#xff0c;无非就是换下节点。 题目 操作给定的二叉树&#xff0c;将其变换为源二叉树的镜像。 数据范围&#xff1a;二叉树的节点数 0 ≤ n ≤ 1000 0≤n≤1000 0≤n≤1000&#xff0c; 二叉树每…

分享几个副业,一天搞100~200不成问题,一不小心收益比你主业还多

每次家庭聚会&#xff0c;总是那些老掉牙的话题在耳边萦绕&#xff1a;“孩子&#xff0c;你工资多少啊&#xff1f;买车买房了吗&#xff1f;”仿佛只有按部就班地上班、结婚生子&#xff0c;才是人生的唯一出路。 然而&#xff0c;在这个充满机遇的时代&#xff0c;谁说“不上…

【Pytorch】2.TensorBoard的运用

什么是TensorBoard 是一个可视化和理解深度爵溪模型的工具。它可以通过显示模型结构、训练过程中的指标和图形化展示训练的效果来帮助用户更好地理解和调试他们的模型 TensorBoard的使用 安装tensorboard环境 在终端使用 conda install tensorboard通过anaconda安装 导入类Sum…

华为ensp中USG6000V防火墙双机热备VRRP+HRP原理及配置

作者主页&#xff1a;点击&#xff01; ENSP专栏&#xff1a;点击&#xff01; 创作时间&#xff1a;2024年5月6日20点26分 华为防火墙双机热备是一种高可用性解决方案&#xff0c;可以将两台防火墙设备组成一个双机热备组&#xff0c;实现主备切换。当主用防火墙出现故障时&…

零基础入门学习Python第二阶01生成式(推导式),数据结构

Python语言进阶 重要知识点 生成式&#xff08;推导式&#xff09;的用法 prices {AAPL: 191.88,GOOG: 1186.96,IBM: 149.24,ORCL: 48.44,ACN: 166.89,FB: 208.09,SYMC: 21.29}# 用股票价格大于100元的股票构造一个新的字典prices2 {key: value for key, value in prices.i…

【强训笔记】day7

NO.1 思路&#xff1a;双指针模拟&#xff0c;begin表示最长数字字符串最后一个字符&#xff0c;而len表示数字字符串的长度&#xff0c;i用来遍历&#xff0c;如果为数字&#xff0c;那么定义j变量继续遍历&#xff0c;直到不为数字&#xff0c;i-j如果大于len&#xff0c;就…

优惠券样式案例

优惠券样式案例 <template><view class"box"><view class"boxItem"><img src"../../../static/come.png" alt"" class"img"/><span class"icon">&#xffe5;</span><s…

cmake进阶:文件操作

一. 简介 前面几篇文章学习了 cmake的文件操作&#xff0c;写文件&#xff0c;读文件。文章如下&#xff1a; cmake进阶&#xff1a;文件操作之写文件-CSDN博客 cmake进阶&#xff1a;文件操作之读文件-CSDN博客 本文继续学习文件操作。主要学习 文件重命名&#xff0c;删…

Flink窗口理论到实践 | 大数据技术

⭐简单说两句⭐ ✨ 正在努力的小叮当~ &#x1f496; 超级爱分享&#xff0c;分享各种有趣干货&#xff01; &#x1f469;‍&#x1f4bb; 提供&#xff1a;模拟面试 | 简历诊断 | 独家简历模板 &#x1f308; 感谢关注&#xff0c;关注了你就是我的超级粉丝啦&#xff01; &a…

vue3中标签的ref属性

组合API-ref属性 在vue2.x中&#xff0c;可以通过给元素添加refxxx属性&#xff0c;然后在代码中通过this.$refs.xxx获取到对应的元素 然而在vue3中时没有$refs这个东西的&#xff0c;因此vue3中通过ref属性获取元素就不能按照vue2的方式来获取。 目标&#xff1a;掌握使用re…

unittest_parameterized批量测试测试用例

import unittest from parameterized import parameterizeddef add(x, y):return xy"""问题&#xff1a;如果有三组数据需要测试&#xff1f;[(1,1,2), (1,2,3), (0,3,3)] """def get_data():return [(1, 2, 3), (3, 0, 3), (2, 1, 3)]# 定义测试…

win10安装DHCP服务--用于2台机器之间搭建简易网络来进入目标机器修改配置

前言&#xff1a; 客户多了&#xff0c;往往会出现各种突发情况。 比如一个客户现场没有DHCP&#xff0c;没有显示器&#xff0c;键盘。 你只有一台笔记本的情况下要配置目标机器的网络。要如何配置&#xff1f;&#xff1f; 这时候就可以使用这篇博客提供的方式了。 Windows…

低功耗UPF设计的经典案列分享

案例1 分享个例子&#xff0c;景芯A72低功耗设计&#xff0c;DBG domain的isolation为何用VDDS_maia_noncpu供电而不是TOP的VDD&#xff1f; 答&#xff1a;因为dbg的上一级是noncpu&#xff0c;noncpu下面分成dbg和两个tbnk。 案例2 景芯A72的低功耗&#xff0c;请问&#…

LangChain-RAG学习之 文档加载器

目录 一、实现原理 二、文档加载器的选择 (一).PDF 加载本地文件 可能需要的环境配置 (二).CSV 1、使用每个文档一行的 CSV 数据加载 CSVLoader 2、自定义 csv 解析和加载 &#xff08;csv_args 3、指定用于 标识文档来源的 列&#xff08;source_column (三)、文件目…

【数据可视化-02】Seaborn图形实战宝典

Seaborn介绍 Seaborn是一个基于Python的数据可视化库&#xff0c;它建立在matplotlib的基础之上&#xff0c;为统计数据的可视化提供了高级接口。Seaborn通过简洁美观的默认样式和绘图类型&#xff0c;使数据可视化变得更加简单和直观。它特别适用于那些想要创建具有吸引力且信…

(数据分析方法)长期趋势分析

目录 一、定义 二、目的 三、方法 1、移动平均法 (1)、简单移动平均法 (2)、加权移动平均法 (3)、指数平滑法 2、最小二乘法 3、线性回归 1、数据预处理 2、观察数据分布建立假设模型 3、定义损失函数 4、批量梯度下降 5、优化 4、LSTM 时序分析 5、特征工程 一…

重写muduo之EPollPoller

1、EPollPoller.h EPollPoller的主要实现&#xff1a;作为poller的派生类&#xff0c;把基类给派生类保留的这些纯虚函数的接口实现出来。 override表示在派生类里面&#xff0c;这些方法是覆盖方法。必须由编译器来保证在基类里面一定有这些函数的接口的声明。在派生类要重写…