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

C++ 合成默认构造函数的真相

对于C++默认构造函数,我曾经有两点误解: 类如果没有定义任何的构造函数,那么编译器(一定会!)将为类定义一个合成的默认构造函数。 合成默认构造函数会初始化类中所有的数据成员。   ...实际上这句话也没有说错,它说明了默认构造函数定义的必要非充分条件,然而却给当时初学C++的我造成了一定的误解。   ...这个问题《Effective C++》并没有给出答案,直到看了《深度探索C++对象模型》,才明白了编译器何时才会帮我们合成一个默认构造函数。   ...我写这篇文章的目的是给和我有同样误解或疑惑的C++初学者看的,如果你对合成默认构造函数已有充分的认识,请忽略本文的内容。 正文 什么是默认构造函数?   ...默认构造函数是可以不用实参进行调用的构造函数,它包括了以下两种情况: 没有带明显形参的构造函数。 提供了默认实参的构造函数。       类设计者可以自己写一个默认构造函数。

81530
  • 您找到你想要的搜索结果了吗?
    是的
    没有找到

    C++中转换构造函数与默认函数的优先级

    前言: 在学习谭浩强《c++面对对象设计》一书中,在学到转换构造函数中, 在转换构造函数中 #include using namespace std; class plural...{ public: plural(int a=0,int b=0):real(a),imaginary(b){}; //默认构造函数 plural(double a){real = a;imaginary...plural类的作用域中有以下定义: plural p1(3) //建立对象p1,由于只有一个参数,调用转换构造函数 那假如我在类中定义了默认构造函数呢,系统如何选择用哪一个函数呢?...结果 在经过反复调式,测验,并结合其他网上文章,得出一下总结 系统是不会运行你函数出现歧义性的,如果你在类中定义了一下: plural(int a=0,int b=0):real(a),imaginary...plural(int a){real = a;imaginary = 3;}; 系统会提示call of overloaded plural(int) is ambiguous 出现歧义性, 在没有歧义性下函数优先级是看

    58530

    C++的四个默认函数(构造函数,析构函数,拷贝函数,赋值函数)

    在C++中,对于一个类,C++的编译器都会为这个类提供四个默认函数,分别是: A() //默认构造函数 ~A() //默认析构函数 A(const A&) //默认拷贝构造函数 A& operator...* a = new Animal(); //将调用默认构造函数 Animal * b = new Animal("花狗"); //将调用自定义的构造函数,对name变量初始化。...//第二种实例化对象的方法 Animal c; //将调用默认构造函数 //注意:对于无参构造函数,不可以使用Animal c(), Animal c("花狗");//将调用自定义构造函数,对name...这个例子调用的是默认的拷贝构造函数(注意看控制台显示,调用了一次构造函数和两次析构函数),可以看出两个对象的成员变量地址是不一样的,当成员变量不存在指针类型是,这样做没什么问题,当类中有指针变量,自动生成的拷贝函数注定会出错...= NULL; } 再运行发现程序崩溃了,调用一次构造函数,调用两次析构函数,两个对象的指针成员所指内存相同,name指针被分配一次内存,但是程序结束时该内存却被释放了两次,导致程序崩溃 ?

    2.3K20

    C++中空类:认识它的6个默认函数和6个构造函数

    C++中空类的6个默认函数默认构造函数:当一个对象被创建但没有被赋予初始值时,会调用默认构造函数。...当我们没有为类定义任何构造函数时,C++会提供一个默认的构造函数,它不接受任何参数,也不执行任何操作。但是,它的存在确保了我们可以创建类的对象。...当我们需要复制一个对象时,拷贝构造函数就会被调用。如果我们没有定义拷贝构造函数,C++会提供一个默认的拷贝构造函数,它会逐个复制对象的所有成员。...如果我们没有定义析构函数,C++会提供一个默认的析构函数,它不执行任何操作,但是它的存在确保了对象可以被正确地销毁。...参数化构造函数:如果你为类定义了参数化构造函数,C++将不再自动提供默认构造函数。如果你仍然需要一个默认构造函数,你必须手动定义它。

    7100

    【C++】类和对象(中):类的默认成员函数,构造函数、析构函数、拷贝构造函数、运算符重载

    1.类的默认成员函数 默认成员函数就是用户没有显示实现,编译器会自动生成的成员函数。...共3点: 1.如果类中没有显示定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦用户显示定义,编译器就不再生成。...这也就是为什么我在这一小节的开头并没有自己写拷贝构造函数的代码但依然可以进行拷贝构造。...,在我们没有自己写拷贝构造函数时,s1确实拷贝给了s2。...3.没有显示实现时,编译器会自动生成一个默认赋值运算符重载,默认赋值运算符重载行为跟默认拷贝构造函数类似。 4.

    11910

    C++从入门到精通——类的6个默认成员函数之构造函数

    前言 类的6个默认成员函数:如果一个类中什么成员都没有,简称为空类。 空类中真的什么都没有吗?并不是,任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。...默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数。 class Date {}; 一、构造函数的概念 C++构造函数是一种特殊的成员函数,用于创建和初始化类的对象。...Date d3(); } 如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义编译器将不再生成。...C2512: “Date”: 没有合适的默认构造函数可用 Date d1; return 0; } 关于编译器生成的默认成员函数,很多读者会有疑惑:不实现构造函数的情况下,编译器会生成默认的构造函数...但是看起来默认构造函数又没什么用? d对象调用了编译器生成的默认构造函数,但是d对象_year/_month/_day,依旧是随机值。也就说在这里编译器生成的默认构造函数并没有什么用?

    17410

    【C++指南】类和对象(二):类的默认成员函数——全面剖析 :构造函数

    这些默认成员函数,包括构造函数、析构函数、拷贝构造函数以及赋值运算符重载函数,是C++类设计中不可或缺的部分,它们定义了对象如何被创建、销毁、复制以及赋值。...取地址运算符重载函数:实际上,C++标准中并没有直接为取地址运算符(&)提供默认的重载机制,因为对象的地址总是由编译器自动处理。...默认构造函数 默认构造函数是没有参数或者所有参数都有默认值的构造函数。如果类中没有显式定义任何构造函数,编译器会自动生成一个默认的无参构造函数。...:我们定义了三个构造函数,其中第一个、第二个以及不写构造函数时编译器默认生成的构造函数,都属于默认构造函数。...而析构函数则在对象的生命周期结束时被调用,确保所有资源得到释放,避免内存泄漏等问题。 结尾 构造函数是C++面向对象编程中的核心概念之一,它决定了对象如何被初始化和配置。

    15410

    C++从入门到精通——类的6个默认成员函数之拷贝构造函数

    拷贝构造函数 前言 一、拷贝构造函数概念 理解 定义 二、拷贝构造函数的特征 三、注意要点 写法 实践 前言 类的6个默认成员函数:如果一个类中什么成员都没有,简称为空类。 空类中真的什么都没有吗?...C++拷贝构造函数是一种特殊的构造函数,用于创建对象时,使用一个已有对象的内容来初始化新的对象。它接受一个同类对象作为参数,并按照该对象的数据成员的值来创建新的对象。...如果没有显式定义拷贝构造函数,编译器会提供一个默认的拷贝构造函数。默认的拷贝构造函数执行的是浅拷贝,即简单地将原对象的值复制给新对象的数据成员。...编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,还需要自己显式实现吗? 当然像日期类这样的类是没必要的。那么下面的类呢?验证一下试试? // 这里会发现下面的程序会崩溃掉?...Data d2 = d1; 实践 如果没有管理资源,一般情况下不需要写拷贝构造函数,默认生成的拷贝构造函数就可以。

    33010

    【C++指南】类和对象(四):类的默认成员函数——全面剖析 : 拷贝构造函数

    了解拷贝构造函数的概念、作用、特点、规则、默认行为以及如何自定义实现,对于编写健壮和高效的C++程序至关重要。...MyClass obj = MyClass(); // 这里的`MyClass()`创建了一个临时对象,然后调用拷贝构造函数赋值给obj 规则 如果类中没有显式定义拷贝构造函数,编译器会提供一个默认的拷贝构造函数...默认拷贝构造函数的行为 默认拷贝构造函数的行为是逐成员复制,对于基本类型成员,直接复制值,也就是浅拷贝;对于对象成员,调用其拷贝构造函数。...自定义拷贝构造函数通过分配新的内存并复制字符串内容,实现了深拷贝。析构函数负责释放动态分配的内存,防止内存泄漏。 总结 拷贝构造函数是C++中用于通过另一个对象初始化新对象的特殊构造函数。...它接受一个同类型的常量引用作为参数。 如果没有显式定义,编译器会提供一个默认的拷贝构造函数,逐成员复制对象。 自定义拷贝构造函数通常用于实现深拷贝,以避免浅拷贝带来的问题。

    11910

    【C++修行之道】类和对象(二)类的6个默认成员函数、构造函数、析构函数

    2.4 一般情况,建议每个类,都可以写一个全缺省的构造(好用) 三、析构函数 3.1 概念 3.2 特性 3.3 C++实现括号匹配和C语言的不同 一、类的6个默认成员函数 如果一个类中什么成员都没有,...2.2.5 自动生成默认构造函数 如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦用户显式定义编译器将不再生成。...C2512: “Date”: 没有合适的默认构造函数可用 Date d1; return 0; } 在这个Date类的定义中,并没有显式定义任何构造函数。...d对象调用了编译器生成的默认构造数,但是d对象_year/_month/_day,依旧是随机值。也就说在这里编译器生成的默认构造函数并没有什么用?...3.3 C++实现括号匹配和C语言的不同 可以明显的看出,C++对应C语言来说简化了不少,对C语言进行了一定的优化。 今天就先到这了!!!

    21210

    【C++】C++ 类中的 this 指针用法 ③ ( 全局函数 与 成员函数 相互转化 | 有参构造函数设置默认参数值 | 返回匿名对象与返回引用 )

    一、全局函数 与 成员函数 相互转化 1、成员函数转为全局函数 - 多了一个参数 C++ 编译器 , 在编译阶段会将 C++ 类的 成员函数 转为 全局函数 , 转换时 , 会 增加一个参数到参数列表开始为止...return s; } 详细代码 , 参考最后的完整代码示例 ; 二、有参构造函数设置默认参数值 ---- 为 Student 类定义了有参构造函数 , 则其默认的无参构造函数 , 就不会生成...执行 Student 的构造函数" << endl; } 此时 , 如果要创建 Student 对象 , 只能调用上述 有参构造函数 , 如果使用 Student s2 的方式调用 默认构造函数 创建...Student 对象 , 就会报错 ; 如下带参数的构造函数 , 并且为其 有参构造函数 的参数 设置一个默认值 , 此时就可以使用 类名 对象名 的方式定义对象变量 ; class Student...创建 Student 实例对象 Student s(18, 180); // 打印对象情况 s.print(); // 调用有参构造函数 , 有参构造函数参数使用默认值 Student s2

    23820

    【C++】深拷贝和浅拷贝 ② ( 默认拷贝构造函数是浅拷贝 | 代码示例 - 浅拷贝造成的问题 )

    一、默认拷贝构造函数是浅拷贝 1、默认拷贝构造函数 如果 C++ 类中 没有定义拷贝构造函数 , C++ 编译器会自动为该类提供一个 " 默认的拷贝构造函数 " , 在函数中对成员变量进行简单的复制操作...; 2、默认拷贝构造函数是浅拷贝机制 C++ 编译器 为 类 自动生成的 默认拷贝构造函数 是 浅拷贝 , 只能拷贝 顶层的 成员变量值 , 如果成员变量 是 引用 或 指针 , 其指向的 类 或 内存空间...对象 , 此时调用的是 拷贝构造函数 , 由于没有定义 拷贝构造函数 , 使用的事 C++ 编译器的 默认拷贝构造函数 , 进行的拷贝 是 浅拷贝 ; 其中的 字符串指针 , 只拷贝了指针的值 , 没有拷贝字符串的具体内容...默认的拷贝构造函数 // C++ 编译器提供的拷贝构造函数 只能进行浅拷贝 Student s2 = s; 二、代码示例 - 浅拷贝造成的问题 下面代码中 , 定义的 Student 类 中 ,...定义了 有参构造函数 和 析构函数 , 没有定义拷贝构造函数 , 因此 C++ 编译器为其生成了 默认拷贝构造函数 , 默认拷贝构造函数 是 浅拷贝 ; 分析下面 创建两个 Student 对象 的代码

    21210

    【C++】多态 ⑧ ( 验证指向 虚函数表 的 vptr 指针 | 对比定义了虚函数的类和没有定义虚函数类的大小 )

    对比 定义了 虚函数 的类 与 没有定义虚函数的类 的大小 , 其它成员都相同 , 定义了虚函数的类多出了 4 字节 , 多出的 4 字节就是 vptr 指针占用的内存空间 ; 一、验证指向 虚函数表...的 vptr 指针 是否存在 1、虚函数表与 vptr 指针由来 " 虚函数表 " 由 C++ 编译器 负责 创建 与 维护 , 被 virtual 关键字 修饰的 虚函数 , 会自动 被 C++ 编译器...中 , 重写了 父类的 virtual 虚函数 , 那么 C++ 编译器会在 子类 虚函数表 中放入该 子类虚函数的 函数指针 ; 如果 C++ 类中存在 virtual 虚函数 , 在创建对象时 ,...; 2、虚函数类与普通函数类对比 - 多出了 vptr 指针的大小 下面的代码中 , 定义了 2 个类 , 区别是 一个定义了 virtual 虚函数 , 另外一个没有定义 虚函数 ; 在 Parent...判断两个类的区别 ; 最终得到 , 有 虚函数 的 类 , 比 没有 虚函数 的 类 , 多 4 字节 , 也就是一个指针的大小 , 定义了 虚函数 的类 , 多出的 4 字节就是 vptr 指针的大小

    22740

    【C++】运算符重载 ⑨ ( 等号 = 运算符重载 | 调用默认浅拷贝构造函数的情况分析 | 等号 = 运算符重载 与 拷贝构造函数 各自使用场景 | 等号 = 操作符重载步骤 )

    , 并支持链式操作 ; 一、等号 = 运算符重载 ---- 1、调用默认浅拷贝构造函数的情况分析 C++ 编译器 为 类 提供的 默认的 拷贝操作 , 是对 成员变量 的简单拷贝 , 是 浅拷贝 ;...C++ 编译器提供的拷贝构造函数 只能进行浅拷贝 Student s2 = s; 在 【C++】深拷贝和浅拷贝 ④ ( 深拷贝示例 ) 博客中实现了 深拷贝构造函数 , 本篇博客主要以该 深拷贝 案例...进行拓展分析 ; 实现了 深拷贝构造函数 后 , 再次使用一个对象为另一个对象赋值时 , 如 Student s2 = s; 代码 , 就会自动调用 深拷贝构造函数 ; 2、等号 = 运算符重载 与...C++ 编译器会自动生成默认的拷贝构造函数 public: // 打印类成员变量 void toString() { cout << "m_age = " << m_age << "...(); // 声明 Student 对象 s2 , 并使用 s 为 s2 赋值 // 该操作会调用 默认的拷贝构造函数 // C++ 编译器提供的拷贝构造函数 只能进行浅拷贝 Student

    27620

    【C++】构造函数调用规则 ( 默认构造函数 | 默认无参构造函数 | 默认拷贝构造函数 | 构造函数调用规则说明 )

    一、默认构造函数 C++ 类中 2 种特殊的构造函数 , 分别是 : 默认无参构造函数 : 如果 C++ 类中 没有定义构造函数 , C++ 编译器会自动为该类提供一个 " 默认的无参构造函数 " ,...没有为 C++ 类定义 构造函数 , C++ 编译器 将自动为该类 生成一个默认的无参构造函数 ; 定义了构造函数 : 如果为 C++ 类 定义了其他类型的构造函数 ( 有参构造函数 / 无参构造函数...在函数中对成员变量进行简单的复制操作 ; 没有定义拷贝构造函数 : 如果 没有为 C++ 类定义 拷贝构造函数 , C++ 编译器 将自动为该类 生成一个 默认的拷贝构造函数 ; 定义了拷贝构造函数..., C++ 编译器不会自动生成 默认的无参构造函数 和 默认的拷贝构造函数 ; 使用 Student s; 代码 创建对象 , 报错 “Student”: 没有合适的默认构造函数可用 ; 说明 C++...默认的无参构造函数 和 默认的拷贝构造函数 ; 使用 Student s; 代码 创建对象 , 报错 “Student”: 没有合适的默认构造函数可用 ; 说明 C++ 编译器没有为该类生成 默认的无参构造函数

    1.3K30

    c++ new和malloc的区别

    A * ptr = new A; A * ptr = (A *)malloc(sizeof(A)); //需要显式指定所需内存大小sizeof(A);  当然了,我这里使用malloc来为我们自定义类型分配内存是不怎么合适的...main() {     A * ptr = (A*)malloc(sizeof(A));     return 0; }  在return处设置断点,观看ptr所指内存的内容:  可以看出A的默认构造函数并没有被调用...,因为数据成员a,b的值并没有得到初始化,这也是上面我为什么说使用malloc/free来处理C++的自定义类型不合适,其实不止自定义类型,标准库中凡是需要构造/析构的类型通通不合适。 ...而使用new来分配对象时:  int main() {     A * ptr = new A; }  查看程序生成的汇编代码可以发现,A的默认构造函数被调用了:  6.对数组的处理  C++提供了new...new_handler是一个指针类型:  namespace std {     typedef void (*new_handler)(); }  指向了一个没有参数没有返回值的函数,即为错误处理函数

    1K00

    细说new与malloc的10点区别

    当时我回答new从自由存储区上分配内存,malloc从堆上分配内存;new/delete会调用构造函数/析构函数对对象进行初始化与销毁;operator new/delete可以进行重载;然后强行分析了一下自由存储区与堆的区别...A * ptr = new A; A * ptr = (A *)malloc(sizeof(A)); //需要显式指定所需内存大小sizeof(A); 当然了,我这里使用malloc来为我们自定义类型分配内存是不怎么合适的...可以看出A的默认构造函数并没有被调用,因为数据成员a,b的值并没有得到初始化,这也是上面我为什么说使用malloc/free来处理C++的自定义类型不合适,其实不止自定义类型,标准库中凡是需要构造/析构的类型通通不合适...而使用new来分配对象时: int main() { A * ptr = new A; } 查看程序生成的汇编代码可以发现,A的默认构造函数被调用了: ?...new_handler是一个指针类型: namespace std { typedef void (*new_handler)(); } 指向了一个没有参数没有返回值的函数,即为错误处理函数。

    1.5K52

    【C++】踏上C++的学习之旅(七):深入“类和对象“世界,掌握编程的黄金法则(二)(内含构造函数和析构函数)

    后面再每个章节中,我会带着大家感受这些默认成员函数的存在) 那这里我们就可以给默认成员函数下一个定义了: 默认成员函数:用户没有显式实现,编译器会自动生成的成员函数称为默认成员函数 这里我先大家简单认识一下这...那我们该怎么解决我是不是会忘记给变量初始化和销毁的问题呢? 此时,构造函数和析构函数就闪亮登场了。...如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,一旦 用户显式定义编译器将不再生成。...C2512: “Date”: 没有合适的默认构造函数可用 Date d1; return 0; } 上面的例子就能很好的解释特征5!...4 构造函数支持函数重载 5 如果类中没有显式定义构造函数,则C++编译器就会自动生成对应的默认构造函数。

    7710
    领券