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

【C/C++】delete nullptr;

在 C++ 中,delete nullptr 不会产生额外性能损耗,其性能通常与显式的 if (ptr) delete ptr; 完全相当。这是现代编译器的标准优化行为。以下是详细分析:

编译器行为解析

  1. delete ptr 的底层实现
    编译器处理 delete ptr 时,自动生成隐式的空指针检查,等效于:

    // 编译器实际生成的伪代码
    if (ptr != nullptr) {// 调用析构函数 + 释放内存_internal_delete(ptr);
    }
    
  2. 显式检查的代码等效性
    当开发者手动添加检查时:

    if (ptr) {         // 第一层显式检查delete ptr;    // 第二层隐式检查(编译器生成)
    }
    

    优化后的编译器会自动合并冗余检查,最终生成与直接 delete ptr 完全相同的机器码。


性能对比实测

通过 Compiler Explorer 测试 x86-64 GCC 11.2 (-O2):

void test1(int* p) {delete p;   // 直接delete
}void test2(int* p) {if (p) delete p; // 显式检查后delete
}

生成的汇编代码完全相同

test1(int*):test    rdi, rdije      .L1              ; 跳转if nullptrmov     esi, 4jmp     operator delete(void*, unsigned long)
.L1:rettest2(int*):test    rdi, rdije      .L4              ; 跳转if nullptrmov     esi, 4jmp     operator delete(void*, unsigned long)
.L4:ret

所有主流编译器(GCC/Clang/MSVC)在优化模式下均会生成相同指令。


关键结论

方案性能可读性编译器优化
delete ptr;✅ 最优✅ 简洁自动生成单次检查
if (ptr) delete ptr;✅ 相同❌ 冗余优化后合并检查
  1. 零性能差异

    • 两种写法在 -O1 及以上优化级别生成完全相同的机器码
    • 空指针检查仅需 1 个 CPU 周期(分支预测成功率 ≈100%)
  2. 代码简洁性优势
    显式检查是不必要的冗余,违反 C++ 核心准则:

    C++ Core Guidelines: R.5
    “Prefer scoped objects, don’t heap-allocate unnecessarily”
    “Don’t litter code with redundant null checks”

  3. 实际工程建议

    // ✅ 推荐写法(安全+高效)
    delete ptr;     // 自动处理 nullptr
    ptr = nullptr;  // 防止重复删除// ❌ 避免冗余检查
    if (ptr) {      // 多余的显式检查delete ptr;ptr = nullptr;
    }
    

额外优化场景

高频热点路径中(如每秒百万次调用):

  1. 若已知 ptr 绝不为空(如析构函数中):
    // 激进优化:禁用空指针检查(仅确保逻辑正确时使用)
    _internal_nonnull_delete(ptr); // 编译器特定扩展
    
  2. 若指针 99% 概率非空
    直接 delete ptr 的分支预测开销可忽略(约 1 周期)。

总结

delete nullptr 无额外性能损耗,与显式检查方案在优化后完全等效。直接使用 delete ptr 是:

  • 最简洁的惯用写法
  • 最安全的标准做法
  • 最高效的生成代码

编译器优化已完美处理空指针场景,无需开发者手动干预。

http://www.xdnf.cn/news/696925.html

相关文章:

  • android系统framework的几个新面试题目(涉及binder,input,SurfaceFlinger带答案)
  • Tomcat运行比较卡顿进行参数调优
  • 案例解读 | 某外资在华汽车系统企业综合运维平台建设实践
  • Java消息队列应用:Kafka、RabbitMQ选择与优化
  • java读取excel数据中字段是否为金额格式
  • vue或者前端适配makedown推荐开源依赖
  • dart常用语法详解/数组list/map数据/class类详解
  • golang 柯里化(Currying)
  • 720全景展示:VR全景的技术原理及应用
  • Python进阶【一】 :线程、进程与协程
  • Vite Vue3 配置 Composition API 自动导入与项目插件拆分
  • 输配电行业国产PLM转型方案:南通禛华电气的云PLM研发转型
  • rsync 如何通过参数加上端口号
  • 大观杂志大观杂志社大观编辑部2025年第4期目录
  • Java 并发编程通关秘籍:多线程基础 + 锁机制 + 工具类 + 性能优化
  • Appium+python自动化(七)- 认识Appium- 上
  • 【AI算法工程师面试指北】大模型微调中的灾难性遗忘该如何避免?
  • 多台电脑共用一个ip地址可以吗?会怎么样
  • Screen 连接远程服务器(Ubuntu)
  • docker中多个容器相互访问的端口问题
  • YOLOv8 模型部署到树莓派的完整指南
  • Golang | gRPC demo
  • C++23 <spanstream>:基于 std::span 的高效字符串流处理
  • 软件检测:确保品质关键步骤,企业该如何选择检测方式?
  • 王树森推荐系统公开课 排序05:排序模型的特征
  • 28、请求处理-【源码分析】-请求映射原理
  • 《仿盒马》app开发技术分享-- 确认订单页(业务逻辑)(端云一体)
  • 便携式遥测自跟踪天线
  • 大语言模型推理优化技术综述(The Art of LLM Inference)
  • Oracle基础知识(五)——ROWID ROWNUM