0

0

C++中的野指针和悬空指针有什么区别?(未初始化与指向已释放内存)

尼克

尼克

发布时间:2026-01-13 13:45:36

|

208人浏览过

|

来源于php中文网

原创

野指针是未初始化的指针,值为栈上随机地址,解引用导致立即崩溃;悬空指针指向已释放内存,可能暂时正常但引发偶发错误。两者均为未定义行为,需用nullptr初始化、智能指针和AddressSanitizer防范。

c++中的野指针和悬空指针有什么区别?(未初始化与指向已释放内存)

野指针是未初始化的指针

野指针指声明后未赋初值的指针,其值是上残留的随机内存地址。它不指向任何合法对象,也不保证可读写,甚至可能指向内核区域导致段错误。C++标准规定使用未初始化指针的行为是undefined behavior,编译器不会报错,但运行时结果不可预测。

常见场景包括:局部指针变量声明后直接解引用、类成员指针未在构造函数中初始化、函数参数指针未检查是否为nullptr就使用。

  • -Wall -Wuninitialized(GCC/Clang)可捕获部分野指针使用,但非万能
  • 现代做法是显式初始化为 nullptr
    int* p = nullptr;
  • RAII 容器如 std::unique_ptr 默认构造即为空,天然规避野指针

悬空指针是指向已释放内存的指针

悬空指针本身可能已被正确初始化并曾合法使用,但其所指对象已被 deletefree() 释放,而指针变量自身未被置空。此时指针仍持有原地址,但该地址对应的内存已归还系统或被复用,再次解引用会触发 use-after-free 错误。

典型诱因有:函数返回局部对象地址、delete 后未置 nullptr、多个指针共享同一堆内存且仅释放一次、容器重分配后迭代器失效仍继续使用。

立即学习C++免费学习笔记(深入)”;

MuleRun
MuleRun

全球首个AI Agent交易平台

下载
  • 释放内存后务必手动置空:
    delete ptr; ptr = nullptr;
  • 优先用智能指针:std::unique_ptr 释放后自动变为空,std::shared_ptr 在最后一个引用销毁时自动释放
  • AddressSanitizer(ASan)可检测大多数悬空指针访问,编译加 -fsanitize=address

两者都会导致 undefined behavior,但触发时机和调试难度不同

野指针出问题往往“立刻见效”——第一次解引用就崩溃(尤其在 Debug 模式下),因为随机地址大概率非法;悬空指针更危险:它可能“暂时正常”,比如释放后的内存尚未被覆盖或重用,程序看似运行无误,直到某次偶然复用该内存才出错,表现为偶发崩溃或数据错乱。

  • 野指针的地址通常远离有效堆/栈范围,OS 更容易拦截
  • 悬空指针的地址仍在进程地址空间内,错误访问可能静默污染其他对象
  • 静态分析工具(如 Clang Static Analyzer)对野指针检出率更高;悬空指针依赖运行时检测(ASan / Valgrind)

如何避免:从习惯到工具链

单靠人工检查不可靠,需结合编码规范与自动化工具。

  • 所有裸指针声明即初始化:int* p = nullptr;,禁止 int* p;
  • 禁用原始 new/delete,改用 std::make_uniquestd::make_shared
  • 容器中避免存储裸指针,改用智能指针或索引/句柄
  • CMake 中启用:set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address -fno-omit-frame-pointer")

最易被忽略的是:析构函数中释放资源后,若该对象还被其他地方持有指针(如观察者列表、缓存表),那些指针就立刻变成悬空指针——这种跨模块生命周期管理,光靠置空解决不了,得靠所有权语义设计。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

534

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

194

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

386

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

569

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

386

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

569

2023.08.10

nginx配置文件详细教程
nginx配置文件详细教程

本专题整合了nginx配置文件相关教程详细汇总,阅读专题下面的文章了解更多详细内容。

0

2026.01.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 0.6万人学习

Rust 教程
Rust 教程

共28课时 | 4.3万人学习

Git 教程
Git 教程

共21课时 | 2.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号