对于基类Base,调用代码可能会尝试通过指向Base的指针销毁派生对象,例如在使用unique_ptr 时。...这种情况导致较早的编码标准对所有基类析构函数都必须是虚拟的提出了全面的要求。这太过分了(即使是常见情况);相反,规则应该是当且仅当基类析构函数是公共的时,才将它们虚函数化。...析构可以看作只是另一种操作,尽管具有使非虚调用变得危险或错误的特殊语义。因此,对于基类析构函数,选择是根据是否允许通过指向Base的指针实际上调用它。“非虚”不是一种选择。...B是可以自己实例化的基类和具体类,因此析构函数必须是公共的,才能创建和销毁B对象。...更好的设计是遵循该产品的建议,为其提供受保护的非虚析构函数。
C.35: A base class destructor should be either public and virtual, or protected and nonvirtual 基类的析构函数要么是公开的虚函数...为了避免无定义的行为。如果析构函数是公有的,那么调用侧的代码就会尝试使用基类指针销毁派生类的对象,在基类的析构函数为非虚函数时其结果时没有定义的。...如果析构函数时保护的,那么调用侧代码就无法通过基类类型指针销毁派生类对象,这是析构函数就没有必要一定是虚函数。析构函数是保护而不是私有的,这样派生类的析构函数才能调用它。...通常,基类的设计者不会知道在析构函数中应该执行什么样的动作。...我们可以想象一种需要保护的虚函数析构函数的情况:当希望允许派生类的对象(只有这个类型)通过基类指针销毁另外一个对象(不是它自己)时。但是我们还没有在实际的开发中遇到这种情况。
派生类的成员或者友元只能通过派生类对象来访问基类的受保护成员,派生类对于一个基类对象中的受保护成员没有任何访问特权 理解最后一条规则可以参考如下例子: class Base { protected:...派生类向基类转换的可访问性 假定D继承B: 只有当D公有地继承B时,用户代码才能使用派生类向基类的转换;如果D继承B的方式是受保护的或者私有的,则用户代码不能使用该转换 无论D以什么方式继承B,D的成员函数和友元都能使用派生类向基类的转换...基类的友元在访问派生类成员时不具有特殊性,同样的派生类的友元也不能随便访问基类的成员。 5....只要基类的析构函数是虚函数,就能确保当我们delete基类指针时将运行正确的析构函数版本。 我们之前介绍过一条经验准则:如果一个类需要析构函数,那么它同样也需要拷贝和赋值操作。...当执行基类的构造函数时,该对象的派生类部分是未被初始化的状态;当执行基类的析构函数时,派生类部分已经被销毁了。
C.127: A class with a virtual function should have a virtual or protected destructor C.127:包含虚函数的类应该有虚析构函数或保护析构函数...包含虚函数的类通常(大多数情况下)通过指向基类的指针使用。通常,最后一个使用者必须通过指向基类的指针调用delete操作,通常是指向基类的智能指针,因此析构函数应该是公开的虚函数。...稍微特殊一些的情况是:如果不希望支持通过指向基类的指针销毁对象,析构函数应该是保护的非虚函数。参见C.35。...包含虚函数的类的析构函数要么是公开的虚函数,要么是保护的非虚函数。...提示针对包含虚函数却没有虚析构函数的类的销毁操作。
使用智能指针可以很大程度上的避免这个问题,因为智能指针就是一个类,当超出了类的作用域是,类会自动调用析构函数,析构函数会自动释放资源。...pa,pb 之间互相引用,两个资源的引用计数为 2,当要跳出函数时,智能指针 pa,pb 析构时两个资源引用计数会减一,但是两者引用计数还是为 1,导致跳出函数时资源没有被释放(A B 的析构函数没有被调用...析构函数 析构函数与构造函数对应,当对象结束其生命周期,如对象所在的函数已调用完毕时,系统会自动执行析构函数。...类析构顺序 派生类本身的析构函数 对象成员析构函数 基类析构函数 因为析构函数没有参数,所以包含成员对象的类的析构函数形式上并无特殊之处。...析构函数与虚函数 析构函数必须是虚函数,因为将可能会被继承的父类的析构函数设置为虚函数,可以保证当我们 new 一个子类,然后使用基类指针指向该子类对象,释放基类指针时可以释放掉子类的空间,防止内存泄漏
合成析构函数:当一个类未定义自己的析构函数时,编译器会为它定义一个合成析构函数。 析构函数体本身并不直接销毁成员。...派生类构造函数: 每个类控制自己的成员的初始化过程。派生类首先初始化基类的部分,然后按照声明的顺序依次初始化派生类的成员。 派生类使用基类的成员: 派生类可以访问基类的公有成员和受保护成员。...不能创建抽象基类的对象。 15.5 访问控制与继承 受保护的成员: 派生类的成员和友元只能访问派生类对象中的基类部分的受保护成员;对于普通的基类对象中的成员不具有特殊的访问权限。...P543 公有、私有和受保护继承: 派生访问说明符对于派生类的成员(及友元)能否访问其直接基类的成员无影响; 对基类成员的访问权限只与基类中的访问说明符有关。...::: 15.7 构造函数与拷贝控制 (1)虚析构函数 在基类中将析构函数定义成虚函数以确保执行正确的析构函数版本。
每个虚继承的子类都有一个虚基类指针(占用一个指针的存储空间,4字节)和虚基类表(不占用类对象的存储空间)当派生类重新定义虚函数时,则将派生类的虚函数的地址添加到虚函数表中。...当一个基类指针指向一个派生类对象时,虚函数表指针指向派生类对象的虚函数表。当调用虚函数时,由于派生类对象重写了派生类对应的虚函数表项,基类在调用时会调用派生类的虚函数,从而产生多态。...虚析构函数:为了防止delete指向派生类对象的基类指针时只调用基类的析构函数引起内存泄漏using namespace std;class Base {public: virtual ~ Base...pb->func(); delete pb; return 0;}运行结果如下:Derived func2delete Deriveddelete Base若Base的析构函数是普通函数...,则delete pb时只会调用Base的析构函数纯虚函数:虚函数声明时候加上=0,包含纯虚函数的类是抽象类,不可实例化,纯虚函数必须被派生类实现。
,由调用者复制删除 //std::unique_ptr被析构时,又会自动对其所指向的对象实施delete //std::unique_ptr被析构时,又会自动对其所指向的对象实施delete class...,基类中必须具备一个虚析构函数 }; //改进的返回值型别 template() 重载 -> 号,当智能指针指向的数据类型为自定义的结构体时,通过 -> 运算符可以获取其内部的指定成员。...operator 重载了 [] 运算符,当 unique_ptr 指针指向一个数组时,可以直接通过 [] 获取指定下标位置处的数据。...并且,B持有的指针不会影响A的引用计数 因此当 std::shared_ptr不再指涉到A时,不会阻止A被析构 */ // 要点速记 // • 使用 std: :weak_ptr 来代替可能空悬的 std
3、构造函数不须要是虚函数,也不同意是虚函数,由于创建一个对象时我们总是要明白指定对象的类型,虽然我们可能通过实验室的基类的指针或引用去訪问它但析构却不一定,我们往往通过基类的指针来销毁对象。...假设基类中采用的是非虚析构函数,当删除基类指针指向的派生类对象时就不会触发动态绑定,因而只会调用基类的析构函数,而不会调用派生类的析构函数。...析构函数没有参数,也没有返回值,而且不能重载,在一个类中只能有一个析构函数。当撤销对象时,编译器也会自动调用析构函数。...,在销毁一个对象时,先调用子类的析构函数,然后再调用基类的析构函数。...假设基类中采用的是非虚析构函数,当删除基类指针指向的派生类对象时就不会触发动态绑定,因而只会调用基类的析构函数,而不会调用派生类的析构函数。
Base *ptr = new Derived(); ptr->who(); // 因为Base有虚析构函数(virtual ~Base() {}),所以 delete 时,会先调用派生类(Derived...)析构函数,再调用基类(Base)析构函数,防止内存泄漏。...; shape2 = nullptr; return 0; } 虚析构函数 虚析构函数是为了解决基类的指针指向派生类对象,并用基类的指针删除派生类对象。...,所以delete释放内存时,先调用子类析构函数,再调用基类析构函数,防止内存泄漏。...若析构函数不可访问,则不能在栈上创建对象。
6、C++中为什么父类要定义虚析构函数(可能看我不太懂C++,问了个奇怪问题) 在C++中,定义虚析构函数(virtual destructor)主要是为了解决多重继承带来的析构问题。...当一个子类被多次继承时,如果在子类的析构函数中没有正确地调用基类的析构函数,就可能导致基类中的资源没有被正确释放,从而引起资源泄漏。...而虚析构函数可以确保在子类的析构函数中正确地调用基类的析构函数,从而避免资源泄漏问题。...具体来说,当一个基类被多次继承时,如果在最顶层的子类的析构函数中没有正确地调用基类的析构函数,就可能导致基类中的资源没有被正确释放。...而如果基类定义了虚析构函数,则在最顶层的子类的析构函数中会自动调用基类的虚析构函数,从而确保基类中的资源被正确释放。
虚析构函数 析构函数的作用是在对象撤销之前做必要的“清理现场”工作;当派生类的对象从内存中撤销时,一般先调用派生类的析构函数,再调用基类的析构函数。...本例中定义了两个类,基类 Base 和派生类 Derived,它们都有自己的构造函数和析构函数。...从运行结果可以看出,执行delete p;语句时只调用了基类的析构函数,却没有调用派生类的析构函数。...当执行delete p;语句时,会先执行派生类的析构函数,再执行基类的析构函数,这样就不存在内存泄露问题了。...通常来说,如果基类中存在一个指向动态分配内存的成员变量,并且基类的析构函数中定义了释放该动态分配内存的代码,那么就应该将基类的析构函数声明为虚函数。
---- 15.5 访问控制与继承 派生类的成员和友元只能访问派生类对象中的基类部分的受保护成员,对于普通的基类对象中的成员不具有特殊的访问权限,即在派生类中也不能通过基类对象来访问基类的 protected...假定 D继承自 B: 只有当 D公有地继承 B时,用户代码才能使用派生类向基类的类型转换;如果 D继承 B的方式是受保护的或私有的,则用户代码不能使用该转换。...析构函数的虚属性会被继承,无论派生类中使用合成的析构函数还是自定义的析构函数,都将是虚函数。这样,就能保证 delete基类指针时总能运行正确的析构函数版本。...如前所述,当一个类中存在拷贝控制成员时,编译器不会为这个类合成移动操作。对于需要定义虚析构函数的基类,也是如此。...当基类构造函数具有默认实参时,实参不会被继承,而是派生类会获得多个继承的构造函数,每个构造函数分别省略掉一个含有默认实参的形参。
C++经常使用RAII来解决上面这种问题,即将资源的生命周期和对象的生命周期进行绑定,对象初始化时资源创建,对象析构时资源销毁。 2....shared_ptr的构造函数可以直接支持传递删除器,实际底层又套了很多的类来解决的,因为他还要将这个删除器传递给析构函数,删除器肯定是在析构函数中使用的嘛,所以还有一个中间传递的过程。...private: //封为私有,不让你随便在任意内存位置创建对象,但没有说不让你创建对象 HeapOnly(){} }; 首先需要知道的一点是,栈和数据段上的对象在其作用域结束或程序终止时,会自动调用析构函数完成对象的资源清理工作...因为封掉析构函数的话,栈和数据段上就无法创建出对象了,因为这些对象无法调用析构函数完成资源清理,所以就只能在堆上创建出对象 了,所以封掉析构函数也是一种只在堆上创建对象的做法。...另一种就是实现一个内部的垃圾回收类GC,用这个类来定义出静态对象_gc,这个_gc是单例类的成员变量,所以当程序结束时,静态对象_gc的生命结束,会自动调用自己的析构函数,而GC本身就是内部类,可以直接访问到静态指针
加了日志后,我们发现当接受一个新连接时: HttpSession 类构造了一次,无析构; HttpConnection 类构造一次,析构一次 断开连接时: HttpSession 类析构一次,然后崩溃。...到这里我们看出,程序的行为已经不符合预期了:接受连接,HttpSession 和 HttpConnection 类应该均构造一次,不会发生析构;连接断开时,HttpSession 和 HttpConnection...类应该均析构一次。...的成员变量智能指针),HttpSession 即使不使用 HttpConnection 对象,在断开连接时,HttpSession 析构会触发其成员变量 HttpConnection 对象的析构,而此时... 的智能指针对象,pConnection 出了 onAccept 函数作用域之后,会自动析构,当析构该对象时,其持有的资源引用计数变为 0,导致 HttpConnection
同时,由于构造函数本身也是一个函数,在函数体内抛出异常将导致当前函数运行结束,并释放已经构造的成员对象,包括其基类的成员,即执行直接基类和成员对象的析构函数。考察如下程序。...由于在类B的构造函数中抛出了异常,而此异常并未在构造函数中被捕捉,所以导致类B的构造函数执行中断,对象b并未构造完成。在类B的构造函数“回滚”的过程中,c的析构函数和类A的析构函数相继被调用。...RAII即资源获取即初始化,也就是说在构造函数中申请分配资源,在析构函数中释放资源。因为C++的语言机制保证了,当一个对象创建的时候,自动调用构造函数,当对象超出作用域的时候会自动调用析构函数。...因此,当构造函数不得已抛出异常时,可以利用“智能指针”unique_ptr来防止内存泄露。...,通过智能指针对内存资源的管理,尽管在类B构造函数抛出异常导致类B析构函数未被执行,但类A的析构函数仍然在对象pA生命周期结束时被调用,避免了资源泄漏。
同时,由于构造函数本身也是一个函数,在函数体内抛出异常将导致当前函数运行结束,并释放已经构造的成员对象,包括其基类的成员,即执行直接基类和成员对象的析构函数。考察如下程序。...由于在类B的构造函数中抛出了异常,而此异常并未在构造函数中被捕捉,所以导致类B的构造函数执行中断,对象b并未构造完成。在类B的构造函数“回滚”的过程中,c的析构函数和类A的析构函数相继被调用。...RAII 即资源获取即初始化,也就是说在构造函数中申请分配资源,在析构函数中释放资源。因为 C++ 的语言机制保证了,当一个对象创建的时候,自动调用构造函数,当对象超出作用域的时候会自动调用析构函数。...因此,当构造函数不得已抛出异常时,可以利用智能指针 unique_ptr 来防止内存泄露。...,通过智能指针对内存资源的管理,尽管在类B构造函数抛出异常导致类B析构函数未被执行,但类 A 的析构函数仍然在对象 pA 生命周期结束时被调用,避免了资源泄漏。
):基类的公用成员和保护成员在派生类中成了私有成员,其私有成员仍为基类私有 受保护的继承(protected inheritance):基类的公用成员和保护成员在派生类中成了保护成员,其私有成员仍为基类私有...在派生时,派生类是不能继承基类的析构函数的,也需要通过派生类的析构函数去调用基类的析构函数。...在派生类中可以根据需要定义自己的析构函数,用来对派生类中所增加的成员进行清理工作;基类的清理工作仍然由基类的析构函数负责。...在执行派生类的析构函数时,系统会自动调用基类的析构函数和子对象的析构函数,对基类和子对象进行清理。...C++明确指出,当derived class 对象经由 base class 指针被删除 而该 base class 带着一个non-virtual 析构函数, 导致对象的 derived 成分没被销毁掉
领取专属 10元无门槛券
手把手带您无忧上云