指针解剖学:穿透C/C++内存操作的核心密码与避坑指南

news2025/2/28 21:40:14

一、指针的本质与内存模型

指针是C/C++的核心特性,本质是内存地址的变量化表示。每个变量在内存中占据连续的字节空间,地址是内存单元的唯一编号(如0x0028FF40)。指针变量存储的是目标数据的首地址,通过地址间接操作数据。

内存模型示例:

int num = 97;      // 假设分配在地址0x0028FF40
int* p = #     // p存储0x0028FF40,*p解引用得到97
  • 步长:指针类型决定地址增减的步长(如int*步长为4字节,char*为1字节)。

二、指针的声明与操作

  1. 声明与初始化

    int* p;          // 声明int型指针
    int a = 10;
    p = &a;         // p指向a的地址
    int* q = NULL;  // 空指针,避免野指针问题
    
  2. 核心操作符

    • &:取地址符,获取变量内存地址(如&a)。
    • *:解引用符,访问指针指向的值(如*p = 20修改a的值)。
  3. 指针运算

    • 算术运算:p++使指针移动到下一个元素地址(步长由类型决定)。
    • 关系运算:比较指针地址大小(需指向同一数组)。

三、多级指针与复杂类型

  1. 二级指针
    指向指针的指针,用于动态内存管理和函数参数传递:

    int pp = &p;  // pp存储p的地址
    pp = 30;      // 修改a的值为30
    
  2. 指针与数组

    • 数组名是首元素地址的常量指针(如int arr; int* p = arr;)。
    • 指针数组 vs. 数组指针:
      int* arr;    // 数组元素为int指针
      int (*pArr); // 指向含5个int元素的数组的指针
      
  3. 函数指针
    指向函数的指针,用于回调机制:

    void (*funcPtr)(int) = &myFunction;  // 声明并赋值
    funcPtr(10);                         // 调用函数
    
  4. 函数指针

    • 本质是函数
    • 就是返回是指针的函数,即返回值是指针,是一个地址
    • 作用代码简洁、一定程度上节约内存
//三种写法都可以
int *fun(int x,int y);
int * fun(int x,int y)int* fun(int x,int y);

四、指针与动态内存管理

  1. 动态分配

    • C:malloc/calloc分配堆内存,需手动释放(free)。
    • C++:new/delete支持类型安全:
      int* arr = new int;  // 动态数组
      delete[] arr;            // 释放内存
      
  2. 智能指针(C++)

    • unique_ptr:独占所有权,禁止拷贝。
    • shared_ptr:引用计数,自动释放内存。

五、高级应用与注意事项

  1. 常量指针 vs. 指针常量

    const int* p1;    // 指向常量的指针(值不可改)
    int* const p2;    // 指针常量(地址不可改)
    
  2. 内存安全

    • 野指针:未初始化或已释放的指针(需置为NULL)。
    • 内存泄漏:忘记释放动态内存(推荐使用RAII机制)。
  3. 与引用的区别
    引用是别名(编译期优化),指针是实体变量,支持运算和多级间接访问。


六、典型应用场景

  1. 数据结构:链表、树的节点链接。
  2. 函数参数:传递大型结构体避免拷贝。
  3. 系统编程:直接操作硬件寄存器或内存映射。

总结

指针是C/C++高效性和灵活性的核心,但也带来内存管理的复杂性。理解其底层机制(地址、步长、类型)和现代实践(智能指针、RAII)是掌握关键。建议结合调试工具(如Valgrind)验证内存操作安全性。

demo

#include <iostream>
using namespace std;
 
// 定义结构体用于示例 
struct Student {
    int age;
    string name;
};
 

int main() {
    // 1. 基础变量指针 
    int a = 10;
    int* p1 = &a;
    *p1 = 20;
    cout << "基础指针: " << a << endl; // 输出 20 [7]()
 
    // 2. 空指针与野指针 
    int* p2 = nullptr; // 安全空指针 
    // int* p3;        // 未初始化野指针(注释避免运行时崩溃)
 
    // 3. 数组指针操作 
    int arr[] = {1,2,3,4,5};
    int* p4 = arr;
    cout << "数组指针: " << *(p4 + 2) << endl; // 输出 3 [3]()
 
    // 4. 字符串指针 
    const char* str = "Hello";
    cout << "字符串指针: " << str[1]  << endl; // 输出 'e' [6]()
 
    // 5. 结构体指针 
    Student s = {20, "Tom"};
    Student* p5 = &s;
    cout << "结构体指针: " << p5->name << endl; // 输出 Tom [5]()
 
    // 6. 动态内存分配 
    int* p6 = new int(100);
    cout << "动态内存: " << *p6 << endl;
    delete p6; // 必须释放 [7]()
 
    // 7. 多维数组指针 
    int matrix[2][3] = {{1,2,3}, {4,5,6}};
    int (*p7)[3]  = matrix;
    cout << "多维数组: " << p7[1][2] << endl; // 输出 6 [3]()
 
    // 8. 函数指针 
    int (*funcPtr)(int, int) = [](int x, int y) { return x + y; };
    cout << "函数指针: " << funcPtr(3,5) << endl; // 输出 8 [6]()
 

    return 0;
}

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

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

相关文章

StableDiffusion打包 项目迁移 项目分发 1

文章目录 SD项目迁移前置知识webui-user.batwebui.batlaunch_utils.py 下一篇开始实践 SD项目迁移 显卡驱动更新&#xff1a;https://www.nvidia.cn/geforce/drivers/ 下载安装三个程序&#xff1a; python3.10.6: https://www.python.org/downloads/release/python-3106/gi…

【数据结构进阶】哈希表

&#x1f31f;&#x1f31f;作者主页&#xff1a;ephemerals__ &#x1f31f;&#x1f31f;所属专栏&#xff1a;数据结构 目录 前言 一、哈希表的概念 二、哈希函数的实现方法 1. 直接定址法 2. 除留余数法 三、哈希冲突 1. 开放定址法&#xff08;闭散列&#xff0…

【蓝桥杯嵌入式】各模块学习总结

系列文章目录 留空 文章目录 系列文章目录前言一、LED模块1.1 赛题要求1.2 模块原理图1.3 编写代码1.4 赛题实战 二、LCD模块2.1 赛题要求2.2 模块原理图2.3 编写代码2.4 赛题实战 三、按键模块3.1 赛题要求3.2 模块原理图3.3 编写代码3.4 赛题实战 四、串口模块4.1 赛题要求4…

Rust学习总结之-枚举

枚举是一个很多语言都有的功能&#xff0c;不过不同语言中其功能各不相同但是要表达的意思是一致的&#xff0c;枚举就是对于一个事物可以穷举出所有可能得值。比如说人的性别就可以用枚举&#xff0c;男人和女人两种。下面我们来学习Rust中的枚举。 一&#xff1a;枚举定义 …

Linux系统管理(十七)——配置英伟达驱动、Cuda、cudnn、Conda、Pytorch、Pycharm等Python深度学习环境

文章目录 前言安装驱动下载安装Cuda编辑环境变量安装Cudnn安装conda验证安装成功配置conda镜像退出conda环境创建python环境查看当前conda环境激活环境安装python包安装pytorch 安装pycharm安装jupyter notebook 前言 深度学习和大语言模型的部署不免会用到Linux系统&#xff…

SLAM算法工程师的技术图谱和学习路径

SLAM(Simultaneous Localization and Mapping)算法工程师是负责开发和实现用于机器人、自动驾驶车辆等领域的SLAM算法的专业人士。下面是SLAM算法工程师需要掌握的基础理论知识: 机器人运动学和动力学:理解机器人在空间中的运动方式和控制方法,包括轮式、蜘蛛腿、飞行器等…

深入了解 Python 中的 MRO(方法解析顺序)

文章目录 深入了解 Python 中的 MRO&#xff08;方法解析顺序&#xff09;什么是 MRO&#xff1f;如何计算 MRO&#xff1f;C3 算法的合并规则C3 算法的合并步骤示例&#xff1a;合并过程解析 MRO 解析失败的场景使用 mro() 方法查看 MRO示例 1&#xff1a;基本用法 菱形继承与…

如何防止 Instagram 账号被盗用:安全设置与注意事项

如何防止 Instagram 账号被盗用&#xff1a;安全设置与注意事项 在这个数字化时代&#xff0c;社交媒体平台如 Instagram 已成为我们日常生活的一部分。然而&#xff0c;随着网络犯罪的增加&#xff0c;保护我们的在线账户安全变得尤为重要。以下是一些关键的安全设置和注意事…

采样算法二:去噪扩散隐式模型(DDIM)采样算法详解教程

参考 https://arxiv.org/pdf/2010.02502 一、背景与动机 去噪扩散隐式模型&#xff08;DDIM&#xff09; 是对DDPM的改进&#xff0c;旨在加速采样过程同时保持生成质量。DDPM虽然生成效果优异&#xff0c;但其采样需迭代数百至数千次&#xff0c;效率较低。DDIM通过以下关键…

各种类型网络安全竞赛有哪些 网络安全大赛的简称

本文是对入门学习的一些概念了解和一些常规场景记录 1.CTF&#xff08;capture the flag&#xff09;是夺旗赛的意思。 是网络安全技术人员之间进行攻防的比赛。 起源1996年DEFCON全球黑客大会&#xff0c;替代之前真实攻击的技术比拼。 (DEFCON极客大会诞生1993&#xff0c;…

包子凑数——蓝桥杯真题Python

包子凑数 输入输出样例 示例 1 输入 2 4 5输出 6样例说明 凑不出的数目包括&#xff1a;1, 2, 3, 6, 7, 11。 示例 2 输入 2 4 6输出 INF样例说明 所有奇数都凑不出来&#xff0c;所以有无限多个 运行限制 最大运行时间&#xff1a;1s最大运行内存: 256M 最大公约数 最大公…

网络通信/IP网络划分/子网掩码的概念和使用

文章目录 概述子网的考题子网掩码的历史有/无类地址子网划分!子网掩码超网技术/CIDR子网掩码和路由IP子网掩码定义 网络规划网络规划-拆子网网络规划-组超网子网划分案例 区分于其他特殊IP地址IP地址和网络地址子网掩码和网络地址子网掩码和广播地址 子网间的通信其他 概述 本…

MySQL--》如何在MySQL中打造高效优化索引

目录 初识索引 索引结构 性能分析 索引使用 最左前缀法则 SQL提示使用 覆盖索引使用 前缀索引使用 索引失效情况 初识索引 索引(index)&#xff1a;是帮助MySQL高效获取数据的数据结构(有序)&#xff0c;在数据之外数据库系统还维护着满足特定查找算法的数据结构&…

盛京开源社区加入 GitCode,书写东北开源生态新篇章

在数字化转型与开源技术蓬勃发展的浪潮下&#xff0c;开源社区已成为推动技术创新的核心力量。盛京开源社区&#xff08;SJOSC&#xff09;作为沈阳地区的开源交流平台&#xff0c;始终致力于连接开发者、企业及高校&#xff0c;构建区域技术生态圈。 现在&#xff0c;盛京开源…

网络运维学习笔记(DeepSeek优化版)005网工初级(HCIA-Datacom与CCNA-EI)链路层发现协议与VLAN技术

文章目录 一、链路层发现协议1.1 思科CDP协议1.2 华为LLDP协议 二、VLAN&#xff08;Virtual Local Area Network&#xff0c;虚拟局域网&#xff09;技术详解2.1 基本概念2.2 技术特性2.3 接口工作原理2.3.1 Access模式2.3.2 Trunk模式 2.4 厂商配置对比思科配置华为配置 2.5 …

DeepSeek开源周Day4:三连发!突破 AI 训练瓶颈的立体解决方案,并行计算三剑客DualPipe、EPLB与Profile-data

项目地址&#xff1a; https://github.com/deepseek-ai/DualPipehttps://github.com/deepseek-ai/eplbhttps://github.com/deepseek-ai/profile-data 开源日历&#xff1a;2025-02-24起 每日9AM(北京时间)更新&#xff0c;持续五天 (4/5)&#xff01; ​ ​ 一、背景概述 …

树莓百度百科更新!宜宾园区业务再添新篇

树莓集团宜宾园区业务不断拓展&#xff0c;主要体现在以下几个方面&#xff1a; 产业布局 -聚焦数字经济核心领域&#xff1a;涵盖软件开发、人工智能、大数据等&#xff0c;吸引众多上下游企业入驻&#xff0c;形成从芯片研发、软件开发到系统集成的完整产业链条。 -推进“双…

RabbitMQ操作实战

1.RabbitMQ安装 RabbitMQ Windows 安装、配置、使用 - 小白教程-腾讯云开发者社区-腾讯云下载erlang&#xff1a;http://www.erlang.org/downloads/https://cloud.tencent.com/developer/article/2192340 Windows 10安装RabbitMQ及延时消息插件rabbitmq_delayed_message_exch…

OpenWebUI配置异常的外部模型导致页面无法打开

一、使用Ollama关闭OpenAI OpenWebUI自带OpenAI的API设置&#xff0c;且默认是打开的&#xff0c;默认情况下&#xff0c;启动后&#xff0c;会不断的去连https://api.openai.com/v1&#xff0c;但是无法连上&#xff0c;会报错&#xff0c;但是不会影响页面&#xff0c;能正常…

鸿蒙兼容Mapbox地图应用测试

鸿蒙Next已经发布一段时间了&#xff0c;很多之前的移动端地图应用&#xff0c;纷纷都要求适配鸿蒙Next。作为开发者都清楚&#xff0c;所谓的适配其实都是重新开发&#xff0c;鸿蒙的开发语言和纯前端的Javascript不同&#xff0c;也可以Android原始开发的语言不同。鸿蒙自带的…