C++对象模型之C++额外成本

embedded/2025/2/26 17:41:35/

1.介绍

        C++与C最大的区别,无疑在于面向对象,面向对象编程给C++带来了强大的特性和灵活性。但同时也带来了一定的运行时和编译时的开销。下面介绍C++对象模型的额外成本及其来源。

2.C++的额外成本

        (1)虚函数和动态多态的成本

        虚函数表(vtable):如果一个类包含虚函数,编译器会给该类生成一个虚函数表,每个对象会包含一个指向虚函数表的指针(vptr),这会增加对象的内存开销。(一个指针额外占用8字节)

        虚函数调用开销:调用虚函数时,需要通过 vptr 查找 vtable,再通过 vtable 找到具体的函数地址。这种间接调用比普通函数调用更慢。

        动态绑定:虚函数支持运行时多态,但这也意味着编译器无法在编译时确定具体调用哪个函数,增加了运行时的开销。

class Base {
public:virtual void foo() { std::cout << "Base::foo()" << std::endl; }
};class Derived : public Base {
public:void foo() override { std::cout << "Derived::foo()" << std::endl; }
};int main() {Base* obj = new Derived();obj->foo();  // 虚函数调用,需要查找 vtable //调用的是派生类的函数delete obj;return 0;
}

        (2)多重继承和虚继承的成本

        多重继承:如果一个类从多个基类继承,对象中会包含每个基类的子对象。这可能导致对象内存布局复杂化,增加内存开销。

        虚继承:虚继承用于解决菱形继承问题,但会引入额外的间接层(通过由虚基类指针实现),增加内存和运行时开销。

class A { int a; };
class B : virtual public A { int b; };
class C : virtual public A { int c; };
class D : public B, public C { int d; };int main() {D obj;std::cout << "Size of D: " << sizeof(obj) << std::endl;  // 可能比预期大return 0;
}

        (3)RTTI(运行时类型识别)成本

        RTTI 允许在运行时获取对象的类型信息,但这需要编译器为每个类生成额外的类型信息,并存储在内存中。使用dynamic_cast时,还需要遍历继承链,增加了运行时开销。

        内存开销:RTTI 会增加程序的内存占用,尤其是对于大型类层次结构。

class Base { virtual void foo() {} };
class Derived : public Base {};int main() {Base* obj = new Derived();if (Derived* d = dynamic_cast<Derived*>(obj)) {std::cout << "Downcast successful" << std::endl;}delete obj;return 0;
}

        (4)异常处理的成本

        异常机制:C++ 的异常处理需要在运行时维护额外的栈帧信息和异常表,这会增加程序的内存和运行时开销。

        性能影响:即使没有抛出异常,异常处理机制也会对性能产生一定影响,尤其是在函数调用和返回时。

void riskyFunction() {throw std::runtime_error("Something went wrong!");
}int main() {try {riskyFunction();} catch (const std::exception& e) {std::cerr << "Caught exception: " << e.what() << std::endl;}return 0;
}

        (5)模版实例化成本

        代码膨胀:模板会在编译时为每种类型生成独立的代码实例,这可能导致生成的目标文件体积增大。

        编译时间:模板的实例化会增加编译时间,尤其是在模板代码复杂或模板参数类型较多时。

template <typename T>
class Box {
public:T value;void set(T v) { value = v; }T get() { return value; }
};int main() {Box<int> intBox;Box<double> doubleBox;return 0;
}

        (6)对象构造和析构的成本

        构造函数和析构函数调用:在复杂的类层次结构中,构造和析构对象可能需要调用多个构造函数和析构函数,增加了运行时开销。

        异常安全:如果构造函数抛出异常,需要确保已分配的资源被正确释放,这会增加额外的逻辑和开销。

class Resource {
public:Resource() { std::cout << "Resource acquired" << std::endl; }~Resource() { std::cout << "Resource released" << std::endl; }
};class Widget {Resource res;
public:Widget() { std::cout << "Widget created" << std::endl; }~Widget() { std::cout << "Widget destroyed" << std::endl; }
};int main() {Widget w;return 0;
}

        (7)内联函数的潜在成本

        代码膨胀:内联函数虽然减少了函数调用的开销,但会将函数体直接插入调用处,可能导致代码体积增大。

        缓存不友好:过度的内联可能导致指令缓存效率降低,影响性能。

inline int add(int a, int b) {return a + b;
}int main() {int result = add(3, 4);return 0;
}

3.总结

        C++对象模型的额外成本主要来自以上七部分。这些特性使C++更加灵活的同时也增加了额外的成本。因此要合理使用C++的特性。

如有错误,敬请指正!!!


http://www.ppmy.cn/embedded/167321.html

相关文章

ESP32移植Openharmony外设篇(8)MQ-3酒精检测传感器

MQ-3酒精检测 模块简介 应用场景 MQ3是MQ传感器系列中最常用的传感器之一。它是金属氧化物半导体&#xff08;MOS&#xff09;类型的传感器。金属氧化物传感器也被称为化学电阻在暴露于醇&#xff0c;因为感测基于所述感测材料的电阻的变化。因此&#xff0c;通过将其放置在…

v4l2子系统学习(五)subdev和media子系统

文章目录 1、声明2、subdev和media子系统2.1、subdev的引入2.2、media子系统的引入2.3、subdev概览和数据结构2.3.1、实例2.3.2、数据结构 2.4、media子系统概览和数据结构2.4.1、拓扑结构2.4.2、数据结构2.4.3、media子系统和subdev的关系 2.5、subdev的注册和使用2.5.1、内核…

Docker Engine stopped

参考【已解决】win10系统 Docker 提示Docker Engine stopped解决全过程记录-CSDN博客 关键的处理办法&#xff1a;需要wsl --update更新一下&#xff0c;然后卸载docker&#xff0c;重新安装后重启系统恢复正常 重启系统后看到running状态

选择排序:简单高效的选择

大家好&#xff0c;今天我们来聊聊选择排序&#xff08;Selection Sort&#xff09;算法。这是一个非常简单的排序算法&#xff0c;适合用来学习排序的基本思路和操作。选择排序在许多排序算法中以其直观和易于实现的特点著称&#xff0c;虽然它的效率不如其他高效算法&#xf…

【Kafka】Windows下安装Kafka(图文记录详细步骤)

【Kafka】Windows下安装Kafka Kafka简介一、Kafka安装前提安装Kafka之前&#xff0c;需要安装JDK、Zookeeper、Scala。1.1、JDK安装&#xff08;version&#xff1a;1.8&#xff09;1.1.1、JDK官网下载1.1.2、JDK网盘下载1.1.3、JDK安装 1.2、Zookeeper安装1.2.1、Zookeeper官网…

计算机考研之数据结构:斐波那契数列专题(1)

不论是在算法还是在编程语言的教材中&#xff0c;都可能会以斐波那契数列为例&#xff0c;或说明其算法上的特点——主要是递归&#xff0c;或说明如何运用某种编程语言编写相应函数。 本文是一篇关于“斐波那契数列”的专题文章&#xff0c;目的是让学习《数据结构》这门课程…

哈希表入门到精通:从原理到 Python 实现全解析

系列文章目录 01-从零开始掌握Python数据结构&#xff1a;提升代码效率的必备技能&#xff01; 02-算法复杂度全解析&#xff1a;时间与空间复杂度优化秘籍 03-线性数据结构解密&#xff1a;数组的定义、操作与实际应用 04-深入浅出链表&#xff1a;Python实现与应用全面解析 …

排序算法适合的场景

排序算法的选择取决于数据规模、特性、稳定性需求、内存限制等因素。以下为常见排序算法及其适用场景&#xff1a; 1. 简单排序算法&#xff08;O(n)&#xff09; 冒泡排序 场景&#xff1a;数据量极小&#xff08;如 n ≤ 100&#xff09;或几乎有序&#xff1b;教学演示。缺点…