首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

为什么我们在C++中没有虚拟构造函数?

在C++中没有虚拟构造函数的原因是因为虚拟函数的主要目的是为了实现多态,而构造函数主要用于初始化对象。当一个对象被创建时,它的类型是确定的,因此在构造函数中不需要虚拟函数的多态特性。

如果在C++中允许虚拟构造函数,那么在调用构造函数时,将会面临一些问题。首先,虚拟函数表是在对象被创建之后才会被初始化的,这意味着在构造函数被调用时,虚拟函数表还没有被初始化,因此无法使用虚拟函数。其次,如果虚拟构造函数被重写,那么在调用构造函数时,应该调用哪个版本的构造函数呢?这将会导致一些混乱和歧义。

因此,为了避免这些问题,C++不允许虚拟构造函数的存在。如果需要实现类似的功能,可以使用工厂模式或者抽象工厂模式来实现。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

C++】多态 ⑨ ( vptr 指针初始化问题 | 构造函数 调用 虚函数 - 没有多态效果 )

, 则 没有 多态效果 ; 一、vptr 指针初始化问题 1、vptr 指针与虚函数表 " 虚函数表 " 由 C++ 编译器 负责 创建 与 维护 , 被 virtual 关键字 修饰的 虚函数 ,..., 对象 被 创建时 , 由 C++ 编译器 对 对象的 vptr 指针进行初始化操作 , 对象 创建完成 后 , 也就是 虚函数 整理完毕 , 全部放到 虚函数后 , vptr 指针 才会指向... 调用 虚函数 , 则 没有 多态效果 ; 父类 的 构造函数 , 调用了 父类的 虚函数 ; 此时 , 如果 创建 子类对象 , 执行 父类构造函数 , 仍然调用 父类 的虚函数 , 子类的虚函数...没有被调用 , 说明 构造函数 执行期间 , 多态没有生效 ; 参考 【C++】继承 ⑧ ( 继承 + 组合 模式的类对象 构造函数 和 析构函数 调用规则 ) 博客构造函数 调用规则 : 构造函数...调用完毕后 , vptr 指针 才指向 父类的 虚函数表 ; 然后 , 调用 子类 的构造函数 , 此时 子类构造函数 调用 fun 虚函数 , 只能调用 子类本身的 fun 函数 , 此时 vptr

21720

C++构造函数初始化列表 ⑤ ( 匿名对象 生命周期 | 构造函数 不能调用 构造函数 )

二、构造函数 调用 构造函数 ---- 1、构造函数 不能调用 构造函数 构造函数 调用 构造函数 是危险行为 ; 构造函数 调用 构造函数 , 并不会为本对象进行初始化操作 ; 构造函数 的调用方式..., 天然就会创建一个 匿名对象 , 如果 没有变量 接收 这个 匿名对象 , 该匿名对象 本行表达式执行完毕后 , 就会被销毁 ; 2、代码示例 - 构造函数调用构造函数 下面的代码 , 先定义...执行 Student 的构造函数" << endl; } 然后 , 无参的 构造函数 , 调用 有参构造函数 ; // 构造函数 调用 构造函数 是危险行为 Student() { /.../ 构造函数调用构造函数 // 此时 创建 的是匿名对象 // 如果 匿名对象 没有被接收 // 就会在 本代码 执行结束后 立刻销毁 Student(18, 180); } Student...endl; } // 构造函数 调用 构造函数 是危险行为 Student() { // 构造函数调用构造函数 // 此时 创建 的是匿名对象 // 如果 匿名对象 没有被接收

18120

c++构造函数学习总结(一)

大家晚上好,今天给大家分享的是c++构造函数,这段时间可能分享c++、Uboot、linux内核的文章会比较多一点,自己的拖延症太强了,得改掉这个坏习惯来。每天进步一点点,日积月累你也是专家。...一、构造函数: 1、什么是构造函数? 关于这个构造函数,简单理解就是一个类,有一个函数,它的函数名称和类名同名,而且这个构造函数没有返回值类型的说法( Test()这个函数就是构造函数了。)...: (1)一般情况下,构造函数定义时自动被调用(主要作用就是自动去初始化类的属性,这个属性通俗一点来说,就是我们所说的变量。...class Test { public: Test(int v) { } }; (2)一个类可以存在多个重载的构造函数...C++重载函数的规则 ——对象定义时会触发构造函数的调用 ——一些情况下可以手动调用构造函数 四、总结: 通过上面的学习,我们大概对构造函数的使用有了一个比较清楚的认识,不过要掌握的更加牢靠,还是通过更多的动手练习哦

56920

C++不要在构造函数和析构函数调用虚函数

虽然可以对虚函数进行实调用,但程序员编写虚函数的本意应该是实现动态联编。构造函数调用虚函数函数的入口地址是在编译时静态确定的,并未实现虚调用。...但是为什么构造函数调用虚函数,实际上没有发生动态联编呢? 1. 不要在构造函数调用虚函数的原因 第一个原因,概念上,构造函数的工作是为对象进行初始化。...当创建某个派生类的对象时,如果在它的基类的构造函数调用虚函数,那么此时派生类的构造函数并未执行,所调用的函数可能操作还没有被初始化的成员,浙江导致灾难的发生。...Visual C++,包含虚函数的类对象的虚指针被安排在对象的起始地址处,并且虚函数表(vtable)的地址是由构造函数写入虚指针的。...从概念上说,析构函数是用来销毁一个对象的,销毁一个对象时,先调用该对象所属类的析构函数,然后再调用其基类的析构函数,所以,调用基类的析构函数时,派生类对象的“善后”工作已经完成了,这个时候再调用在派生类定义的函数版本已经没有意义了

2.7K30

C++同时存在继承以及组合时候,构造函数构造顺序

C++的一大特点就是面向对象,面向对象主要就是类的一些相关特性(封装、继承、多态)。 那么类的继承以及类的成员属性包含其他类实例对象的时候,构造函数构造顺序到底是怎么样子的呢?...那么当一个类对象既包含了继承关系同时也自身的成员属性包含了其他类对象的实例化的时候,那么这时候实例化该类的对象时候,构造函数的顺序会是怎么样子的呢?下面来看看这一段代码吧。...// 实例化一个C类的对象 system("pause"); return 0; } 这就是一个简单的继承加上组合的小demo,从上面的小demo可以看出类C继承与类A,并且类C组合了类B的实例化对象...,那么我们可以直接到以下的结果,可以得知。...A 类的构造函数 B 类的构造函数 C 类的构造函数 构造顺序是首先构造继承的父类,其次构造组合的实例对象,最后才是构造自己本身。

1.1K20

C++:39---继承构造函数、析构函数的关系

一、继承构造函数的关系 如果父类没有构造函数,则子类初始化时不需要构造父类 如果父类有构造函数,则子类初始化自己的构造函数时,要先初始化父类的构造函数 基类的构造函数必须在派生类的构造函数初始化列表来进行初始化...总结:构造自己(子类)之前,需要先构造父类 演示案例 例如:下面的父类A有构造函数,则子类B初始化构造函数时,必须要构造父类A class A //父类 { int a_data; public:...b_data = data; } ~B() {} }; 二、若一个类定义了另一类的构造函数关系 与继承构造父类的构造函数相类似: 如果类定义的对象没有构造函数,则该类初始化时不需要构造该对象的构造函数...如果类定义的对象有构造函数,则该类初始化自己的构造函数时,要先初始化该对象的构造函数 总结:构造自己之前,需要先构造类内的其他对象 注意事项: 类定义的其它类对象必须在构造函数的初始化列表初始化...{ b_data = data; } ~B() {} }; 三、继承父、子类的构造函数、析构函数的执行顺序 构造函数执行顺序: 第一步:先构造父类的构造函数 第二步:如果类定义了其他类的对象,再初始化其他类的构造函数

97620

为什么应该尽可能避免静态构造函数初始化静态字段?

不同的是Foo以内联(inline)赋值的方法进行初始化,而Bar则将初始化操作定义静态构造函数。...如下所示的两段IL代码分别来源于Foo和Bar,我们可以看到虽然Foo类没有显式定义静态构造函数,但是编译器会创建一个默认的静态构造函数,针对静态字段的初始化就放在这里。...我们可以进一步看出,自动生成的这个静态构造函数我们自己写的并没有本质的不同。...但是当我们调用一个并不涉及类型静态字段的Invoke方法时,定义Foo的静态构造函数会自动执行,但是定义Bar的则不会,由此可以看出一个类型的静态构造函数的执行时机与类型是否具有beforefieldinit...四、关于“All-Zero”结构体 如果我们一个结构体显式定义了一个静态构造函数,当我们调用其构造函数之前,静态构造函数会自动执行。

16110

C++虚拟函数的内存分配机制

因为虚拟函数的地址翻译取决于对象的内存地址,而不取决于数据类型(编译器对函数 调用的合法性检查取决于数据类型)。...原来,如果类定义了虚拟函数,该类及其派生类 就要生成一张虚函数表,即vtable。而在类的对象地址空间中存储一个该虚函数表的入口, 占4个字节,这个入口地址是构造对象是由编译器写入的。...有如下C++程序: //#include #include using namespace std; class CMem { public: CMem...,由于对象的内存空间中包含了虚函数表的入口, 编译器能够由这个入口找到适当的虚函数,这个函数的地址不再由数据类型决定了。...到此,虚函数的秘密终于大白于天下了。虚函数C++语法的重点和难点。

95320

C++构造函数和析构函数抛出异常的注意事项

1.构造函数抛出异常 C++构造函数,既需要分配内存,又需要抛出异常时要特别注意防止内存泄露的情况发生。...因为构造函数抛出异常,概念上将被视为该对象没有被成功构造,因此当前对象的析构函数就不会被调用。...由于类B的构造函数抛出了异常,而此异常并未在构造函数中被捕捉,所以导致类B的构造函数的执行中断,对象b并未构造完成。类B的构造函数“回滚”的过程,c的析构函数和类A的析构函数相继被调用。...最后,由于b并没有被成功构造,所以main()函数结束时,并不会调用b的析构函数,也就很容易造成内存泄露。 2.析构函数抛出异常 析构函数是可以抛出异常的,但是这样做很危险,请尽量不要这要做。...} } 面对析构函数抛出异常时,程序猿要注意以下几点: (1)C++析构函数的执行不应该抛出异常; (2)假如析构函数抛出了异常,那么你的系统将变得非常危险,也许很长时间什么错误也不会发生

2K10

C++】继承 ⑦ ( 继承的对象模型分析 | 继承构造函数和析构函数 )

, 除了继承自 B 类的 int x 和 int y 成员 , 还有一个自己的 int z 成员 , 在内存是 3 个 int 类型的空间 ; 3、问题引入 - 派生类对象构造函数和析构函数调用...上述 继承 的过程 , 每一层继承 , 都继承了上一级 父类的 成员变量 , 同时自己也定义了新的成员变量 ; 派生类对象 构造时 , 构造函数如何进行调用 ; 派生类对象 析构时 , 析构函数如何进行调用...---- 1、子类构造函数与析构函数调用顺序 继承构造函数和析构函数 : 子类构造 : 子类对象 进行 构造 时 , 需要调用 父类 的 构造函数 对 继承自父类的 成员变量 进行 初始化 操作...; 然后 , 再调用 父类 的 析构函数 , 析构 继承自父类的成员 ; 2、子类构造函数参数列表 如果 父类 的 构造函数 有 参数 , 则 需要再 子类 的 初始化列表 显示调用 该有参构造函数...A { public: B(int b) { this->y = b; cout << "B 构造函数调用" << endl; } } 如果 A 类 没有 默认的构造函数 , B 类的构造函数

19440

C++核心准则C.82:不要在构造函数或析构函数调用虚函数

C.82: Don't call virtual functions in constructors and destructors C.82:不要在构造函数或析构函数调用虚函数 Reason...到目前为止,被调用的函数应该只属于构造对象本身,而不是可能存在于派生类的某个覆盖函数。那样做非常难理解。...最坏的情况,构造函数或者析构函数中直接或间接调用一个没有实现的纯虚函数会导致没有定义的行为。...从构造函数和析构函数调用虚函数并不是本身有什么错误。这种调用的语义是安全的。然而,经验表明这样的调用很少是必须的,很容易扰乱维护者,如果被新手使用会成为错误源。...提示来自构造函数或析构函数的虚函数调用。

74450

构造函数以及析构函数PHP需要注意的地方

构造函数以及析构函数PHP需要注意的地方 基本上所有的编程语言类中都会有构造函数和析构函数的概念。...没事,我们一个一个来看: 子类如果重写了父类的构造或析构函数,如果不显式地使用parent::__constuct()调用父类的构造函数,那么父类的构造函数不会执行,如C类 子类如果没有重写构造或析构函数...引用如果没有释放,析构函数是不会执行的。 构造函数的低版本兼容问题 PHP5以前,PHP的构造函数是与类名同名的一个方法。...如果将构造函数设置成非公共的,那么你将无法实例化这个类。这一点单例模式被广泛应用,下面我们直接通过一个单例模式的代码看来。...关于单例模式为什么要让外部无法实例化的问题,我们可以看看之前的设计模式系统文章的单例模式。

1.6K20

《挑战30天C++入门极限》C++利用构造函数与无名对象简化运算符重载函数

C++利用构造函数与无名对象简化运算符重载函数   完整描述思想之前,我们先看一下如下的例子,这个例子的加运算符重载是以非成员函数的方式出现的: //程序作者:管宁 //...,但仔细看看的确也情理,参数顺序改变后c++无法识别可供使用的运算符重载函数了。   ...a = a + 10;//正确 a = 10 + a;//正确 cout<<a.a<<endl; system("pause"); }   代码我们使用内联函数的目的是为了缩减开销...,但事实上我们仍然觉得是比较麻烦的,例子的情况都还是非成员函数的情况,如果运算符重载函数是作为类成员函数,那么问题就来了,重载函数的第一个参数始终被隐藏,我们无发让int形参排列隐藏参数的前面,从而导致...答案是否定的,我们可以利用类构造函数对参与运算的整型对象进行显式的类型转换,从而生成无名对象参与同类型对象的加运算,这样做可以缩减代码量,提高程序的可读性。

44620

C++构造函数初始化列表 ③ ( 构造函数 的 初始化列表 为 const 成员变量初始化 )

初始化 , 所有的构造函数都要进行初始化操作 ; 一、构造函数 的 初始化列表 为 const 成员变量初始化 1、初始化 const 常量成员 如果 类 定义了 被 const 修饰 的 成员变量...进行赋值 的 , 因此 这里 必须在 构造函数的 初始化列表 对 const 成员变量 进行赋值 ; 2、错误代码示例 - 没有初始化常量成员 在下面的 类 B , 定义了常量 const int...========== 生成: 成功 0 个,失败 1 个,最新 0 个,跳过 0 个 ========== 在编译时 , 所有的 构造函数 上 , 都会报错 ; 所有的 构造函数 , 都必须 初始化列表..., 对 常量成员 进行初始化操作 ; 3、正确代码示例 - 初始化列表初始化常量成员 在下面的 类 B , 所有的 构造函数 , 都要使用 初始化列表 初始化 常量成员 , 只要遗漏一个构造函数...类型成员变量 const int m_const_int; // 常量成员 }; int main() { // 通过 B 的有参构造函数 // 其中 构造函数的参数 作为 参数列表 的参数值

17530
领券