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

C++代码优化--要求或禁止在堆中产生对象

目录

1.引言

2.栈与堆区别

2.1. 栈(Stack)

2.2. 堆(Heap)

3.限制在堆上分配内存的好处

4.对象在栈上分配内存的方法

4.1. 使用RAII(资源获取即初始化)

4.2. 避免使用new和delete

4.3. 限制对象的生命周期

4.禁止在堆上分配对象的方法

4.1. 将构造函数设为私有或者受保护

4.2. 删除operator new和operator delete

4.3. 使用工厂函数控制对象创建

4.4. 模板技术限制堆分配


1.引言

        在C++代码优化中,内存分配策略(即对象是在栈上还是在堆上分配)对程序的性能和资源管理有着显著影响。合理地要求或禁止在堆中产生对象,可以有效提高程序的效率、减少内存碎片,并增强代码的可维护性。本文将详细探讨如何在C++中要求或禁止对象在堆中分配,并介绍相关的优化技术和最佳实践。

2.栈与堆区别

2.1. 栈(Stack)

  • 分配方式:由编译器自动管理,分配和释放速度极快。

  • 生命周期:对象的生命周期由其作用域决定,超出作用域自动销毁。

  • 大小限制:栈的大小通常较小,过多或过大的栈分配可能导致栈溢出。

  • 访问速度:访问速度快,缓存命中率高。

2.2. 堆(Heap)

  • 分配方式:由程序员手动管理(使用newdelete),也可通过智能指针管理。

  • 生命周期:对象的生命周期由程序员控制,需要手动释放,否则会导致内存泄漏。

  • 大小限制:堆的大小远大于栈,适合分配大对象或数量众多的对象。

  • 访问速度:访问速度相对较慢,且可能导致内存碎片。

3.限制在堆上分配内存的好处

  1. 性能提升:栈分配速度更快,减少动态内存分配的开销。

  2. 内存管理简化:避免手动管理堆内存,减少内存泄漏和悬挂指针的风险。

  3. 缓存优化:栈上对象更容易被CPU缓存命中,提高访问效率。

  4. 资源控制:限制堆分配可以防止程序过度消耗内存资源,提升稳定性。

4.对象在栈上分配内存的方法

4.1. 使用RAII(资源获取即初始化)

RAII是一种C++编程习惯,通过将资源的获取与对象的生命周期绑定,确保资源在对象的构造和析构中被正确管理。这样可以减少对堆分配的需求。

#include <iostream>
#include <vector>class Resource {
public:Resource() { std::cout << "Resource acquired\n"; }~Resource() { std::cout << "Resource released\n"; }void doSomething() { std::cout << "Doing something\n"; }
};void func() {Resource res; // 栈上分配res.doSomething();
} // 自动调用析构函数,释放资源int main() {func();return 0;
}

4.2. 避免使用new和delete

尽量在需要的地方使用栈对象,避免动态分配。使用智能指针(如std::unique_ptrstd::shared_ptr)来管理堆对象,但在可能的情况下,优先使用栈对象。

#include <memory>class MyClass {
public:void doWork() {}
};int main() {MyClass obj; // 栈上分配obj.doWork();// 使用智能指针管理堆对象std::unique_ptr<MyClass> ptr = std::make_unique<MyClass>();ptr->doWork();return 0;
}

4.3. 限制对象的生命周期

通过设计类的接口和生命周期管理,确保对象在栈上分配,并且在作用域结束时自动销毁。

#include <iostream>class ScopedObject {
public:ScopedObject() { std::cout << "ScopedObject created\n"; }~ScopedObject() { std::cout << "ScopedObject destroyed\n"; }void doSomething() { std::cout << "ScopedObject doing something\n"; }
};void process() {ScopedObject obj; // 栈上分配obj.doSomething();
} // 自动销毁int main() {process();return 0;
}

4.禁止在堆上分配对象的方法

4.1. 将构造函数设为私有或者受保护

通过将构造函数设为私有或受保护,可以防止外部直接使用new进行对象创建,仅允许在特定的上下文中创建对象,如工厂函数或友元类。

#include <iostream>class NoHeap {
public:static NoHeap create() {return NoHeap(); // 通过静态成员函数创建对象}void doWork() { std::cout << "NoHeap doing work\n"; }private:NoHeap() { std::cout << "NoHeap constructed\n"; }~NoHeap() { std::cout << "NoHeap destructed\n"; }
};int main() {NoHeap obj = NoHeap::create(); // 只能在栈上分配obj.doWork();// 以下代码将无法编译,因为构造函数是私有的// NoHeap* p = new NoHeap(); // 编译错误return 0;
}

4.2. 删除operator new和operator delete

通过在类中删除operator newoperator delete,可以彻底禁止通过newdelete进行堆分配。

#include <iostream>class NoHeapAlloc {
public:NoHeapAlloc() { std::cout << "NoHeapAlloc constructed\n"; }~NoHeapAlloc() { std::cout << "NoHeapAlloc destructed\n"; }void doWork() { std::cout << "NoHeapAlloc doing work\n"; }// 删除全局和数组版本的operator new和operator deletevoid* operator new(size_t) = delete;void operator delete(void*) = delete;void* operator new[](size_t) = delete;void operator delete[](void*) = delete;
};int main() {NoHeapAlloc obj; // 栈上分配obj.doWork();// 以下代码将无法编译,因为operator new被删除// NoHeapAlloc* p = new NoHeapAlloc(); // 编译错误return 0;
}

4.3. 使用工厂函数控制对象创建

通过工厂函数,仅允许在栈上创建对象,并在工厂函数中返回对象的副本,而不是指针。

#include <iostream>class FactoryControlled {
public:void doWork() { std::cout << "FactoryControlled doing work\n"; }private:// 构造函数私有,防止外部直接创建FactoryControlled() { std::cout << "FactoryControlled constructed\n"; }~FactoryControlled() { std::cout << "FactoryControlled destructed\n"; }// 工厂函数为友元friend FactoryControlled createFactoryControlled();
};// 工厂函数
FactoryControlled createFactoryControlled() {return FactoryControlled();
}int main() {FactoryControlled obj = createFactoryControlled(); // 只能通过工厂函数创建obj.doWork();// 以下代码将无法编译,因为构造函数是私有的// FactoryControlled* p = new FactoryControlled(); // 编译错误return 0;
}

4.4. 模板技术限制堆分配

通过模板编程技术,限制特定类只能在栈上分配。例如,使用CRTP(Curiously Recurring Template Pattern)模式,或者在基类中禁用operator new

#include <iostream>template <typename T>
class StackOnly {
public:void doWork() { std::cout << "StackOnly doing work\n"; }private:// 禁用operator new和operator deletevoid* operator new(size_t) = delete;void operator delete(void*) = delete;
};class MyStackOnly : public StackOnly<MyStackOnly> {
public:MyStackOnly() { std::cout << "MyStackOnly constructed\n"; }~MyStackOnly() { std::cout << "MyStackOnly destructed\n"; }
};int main() {MyStackOnly obj; // 栈上分配obj.doWork();// 以下代码将无法编译,因为operator new被禁用// MyStackOnly* p = new MyStackOnly(); // 编译错误return 0;
}

结合这些方法,可以编写高效、安全且易于维护的C++代码,特别适用于对性能和资源管理有严格要求的应用场景,如游戏开发、高性能服务器和嵌入式系统等。

http://www.lryc.cn/news/475189.html

相关文章:

  • MybatisPlus入门(六)MybatisPlus-空值处理
  • 钉钉内集成第三方免密登录(Vue+.Net)
  • 卷积神经网络实验三:模型优化(1)
  • STM32F103的CAN通讯接收测试
  • 【Rust中的智能指针】
  • 基于深度学习的社交网络中的社区检测
  • 【Python基础】
  • 【玉米叶部病害识别】Python+深度学习+人工智能+图像识别+CNN卷积神经网络算法+TensorFlow
  • 【设计模式】如何用C++实现依赖倒置
  • 使用onnxruntime-web 运行yolov8-nano推理
  • Gin框架html/vue前端使用hls.js播放/点播m3u8(hls)格式视频
  • HarmonyOS 私仓搭建
  • Mybatis学习笔记(二)
  • Google“Big Sleep“人工智能项目发现真实软件漏洞
  • npm入门教程5:package.json
  • docker-高级(待补图)
  • Qt 文件目录操作
  • Pandas 数据清洗
  • IO学习笔记
  • 汇编练习-1
  • 初识二叉树( 二)
  • AcWing1077-cnblog
  • 五、SpringBoot3实战(1)
  • 练习LabVIEW第三十三题
  • 如何在服务器端对PDF和图像进行OCR处理
  • Windows 下实验视频降噪算法 MeshFlow 详细教程
  • Python入门:如何正确的控制Python异步并发量(制并发量的关键技巧与易错点解析)
  • qt QCheckBox详解
  • PAT甲级-1041 Be Unique
  • 【jvm】如何设置堆内存大小