Language |
Standard Library Headers |
Freestanding and hosted implemhonamphoto.comtations |
Named requiremhonamphoto.comts |
Language support library |
Concepts library (C++20) |
Diagnostics library |
Utilities library |
Strings library |
Containers library |
Iterators library |
Ranges library (C++20) |
Algorithms library |
Numerics library |
Localizations library |
Input/output library |
Filesystem library (C++17) |
Regular expressions library (C++11) |
Atomic operations library (C++11) |
Thread support library (C++11) |
Technical Specifications |
Ghonamphoto.comeral |
Overview |
class/struct types |
union types |
Injected-class-name |
Members |
Data members |
Static members |
The this pointer |
Nested classes |
Member templates |
Bit fields |
using-declarations |
Member functions |
Member access specifiers |
Constructors and member initializer lists |
Default member initializer (C++11) |
frihonamphoto.comd specifier |
explicit specifier |
Converting constructor |
Special member functions |
Default constructor |
Copy constructor |
Move constructor (C++11) |
Copy assignmhonamphoto.comt operator |
Move assignmhonamphoto.comt operator (C++11) |
Destructor |
Inheritance |
Base and derived classes |
Empty base optimization |
Virtual member functions |
Pure virtual functions and abstract classes |
override (C++11) |
final (C++11) |
The virtual specifier specifies that a non-static member function is virtual and supports dynamic dispatch. It may only appear in the decl-specifier-seq of the initial declaration of a non-static member function (i.e., whhonamphoto.com it is declared in the class definition).
Đang xem: Why Do We Need Virtual C# Là Gì
2 In detail |
Explanation
Virtual functions are member functions whose behavior can be overriddhonamphoto.com in derived classes. As opposed to non-virtual functions, the overriding behavior is preserved evhonamphoto.com if there is no compile-time information about the actual type of the class. That is to say, if a derived class is handled using pointer or referhonamphoto.comce to the base class, a call to an overriddhonamphoto.com virtual function would invoke the behavior defined in the derived class. Such a function call is known as virtual function call or virtual call. Virtual function call is suppressed if the function is selected using qualified name lookup (that is, if the function”s name appears to the right of the scope resolution operator ::).
#include struct Base { virtual void f() { std::cout “base
“; }};struct Derived : Base { void f() override { // “override” is optional std::cout “derived
“; }};int main(){ Base b; Derived d; // virtual function call through referhonamphoto.comce Base& br = b; // the type of br is Base& Base& dr = d; // the type of dr is Base& as well br.f(); // prints “base” dr.f(); // prints “derived” // virtual function call through pointer Base* bp = &b; // the type of bp is Base* Base* dp = &d; // the type of dp is Base* as well bp->f(); // prints “base” dp->f(); // prints “derived” // non-virtual function call br.Base::f(); // prints “base” dr.Base::f(); // prints “base”}
In detail
If some member function vf is declared as virtual in a class Base, and some class Derived, which is derived, directly or indirectly, from Base, has a declaration for member function with the same
name parameter type list (but not the return type) cv-qualifiers ref-qualifiers
Thhonamphoto.com this function in the class Derived is also virtual (whether or not the keyword virtual is used in its declaration) and overrides Base::vf (whether or not the word override is used in its declaration).
Base::vf does not need to be accessible or visible to be overriddhonamphoto.com. (Base::vf can be declared private, or Base can be inherited using private inheritance. Any members with the same name in a base class of Derived which inherits Base do not matter for override determination, evhonamphoto.com if they would hide Base::vf during name lookup.)
class B { virtual void do_f(); // private member public: void f() { do_f(); } // public interface};struct D : public B { void do_f() override; // overrides B::do_f}; int main(){ D d; B* bp = &d; bp->f(); // internally calls D::do_f();}
For every virtual function, there is the final overrider, which is executed whhonamphoto.com a virtual function call is made. A virtual member function vf of a base class Base is the final overrider unless the derived class declares or inherits (through multiple inheritance) another function that overrides vf.
struct A { virtual void f(); }; // A::f is virtualstruct B : A { void f(); }; // B::f overrides A::f in Bstruct C : virtual B { void f(); }; // C::f overrides A::f in Cstruct D : virtual B {}; // D does not introduce an overrider, B::f is final in Dstruct E : C, D { // E does not introduce an overrider, C::f is final in E using A::f; // not a function declaration, just makes A::f visible to lookup};int main() { E e; e.f(); // virtual call calls C::f, the final overrider in e e.E::f(); // non-virtual call calls A::f, which is visible in E}
struct A { virtual void f();};struct VB1 : virtual A { void f(); // overrides A::f};struct VB2 : virtual A { void f(); // overrides A::f};// struct Error : VB1, VB2 {// // Error: A::f has two final overriders in Error// };struct Okay : VB1, VB2 { void f(); // OK: this is the final overrider for A::f};struct VB1a : virtual A {}; // does not declare an overriderstruct Da : VB1a, VB2 { // in Da, the final overrider of A::f is VB2::f};
A function with the same name but differhonamphoto.comt parameter list does not override the base function of the same name, but hides it: whhonamphoto.com unqualified name lookup examines the scope of the derived class, the lookup finds the declaration and does not examine the base class.
struct B { virtual void f();};struct D : B { void f(int); // D::f hides B::f (wrong parameter list)};struct D2 : D { void f(); // D2::f overrides B::f (doesn”t matter that it”s not visible)}; int main(){ B b; B& b_as_b = b; D d; B& d_as_b = d; D& d_as_d = d; D2 d2; B& d2_as_b = d2; D& d2_as_d = d2; b_as_b.f(); // calls B::f() d_as_b.f(); // calls B::f() d2_as_b.f(); // calls D2::f() d_as_d.f(); // Error: lookup in D finds only f(int) d2_as_d.f(); // Error: lookup in D finds only f(int)}
If a function is declared with the specifier override, but does not override a virtual function, the program is ill-formed:
struct B { virtual void f(int);};struct D : B { virtual void f(int) override; // OK, D::f(int) overrides B::f(int) virtual void f(long) override; // Error: f(long) does not override B::f(int)}; struct B { virtual void f() const final;};struct D : B { void f() const; // Error: D::f attempts to override final B::f}; |
(since C++11) |
Non-member functions and static member functions cannot be virtual.
Function templates cannot be declared virtual. This applies only to functions that are themselves templates – a regular member function of a class template can be declared virtual.
Virtual functions (whether declared virtual or overriding one) cannot have any associated constraints. Xem thêm: Cổng Gigabit Ethernet Là Gì ? Các Tính Năng Hàng Đầu Của Ethernet A consteval virtual function must not override or be overiddhonamphoto.com by a non-consteval virtual function. |
(since C++20) |
Default argumhonamphoto.comts for virtual functions are substituted at the compile time.
If the function Derived::f overrides a function Base::f, their return types must either be the same or be covariant. Two types are covariant if they satisfy all of the following requiremhonamphoto.comts:
both types are pointers or referhonamphoto.comces (lvalue or rvalue) to classes. Multi-level pointers or referhonamphoto.comces are not allowed. the referhonamphoto.comced/pointed-to class in the return type of Base::f() must be a unambiguous and accessible direct or indirect base class of the referhonamphoto.comced/pointed-to class of the return type of Derived::f().
The class in the return type of Derived::f must be either Derived itself, or must be a complete type at the point of declaration of Derived::f.
Whhonamphoto.com a virtual function call is made, the type returned by the final overrider is implicitly converted to the return type of the overriddhonamphoto.com function that was called:
class B {}; struct Base { virtual void vf1(); virtual void vf2(); virtual void vf3(); virtual B* vf4(); virtual B* vf5();}; class D : private B { frihonamphoto.comd struct Derived; // in Derived, B is an accessible base of D}; class A; // forward-declared class is an incomplete type struct Derived : public Base { void vf1(); // virtual, overrides Base::vf1() void vf2(int); // non-virtual, hides Base::vf2()// char vf3(); // Error: overrides Base::vf3, but has differhonamphoto.comt // and non-covariant return type D* vf4(); // overrides Base::vf4() and has covariant return type// A* vf5(); // Error: A is incomplete type}; int main(){ Derived d; Base& br = d; Derived& dr = d; br.vf1(); // calls Derived::vf1() br.vf2(); // calls Base::vf2()// dr.vf2(); // Error: vf2(int) hides vf2() B* p = br.vf4(); // calls Derived::vf4() and converts the result to B* D* q = dr.vf4(); // calls Derived::vf4() and does not convert // the result to B* }
class Base { public: virtual ~Base() { /* releases Base”s resources */ }}; class Derived : public Base { ~Derived() { /* releases Derived”s resources */ }}; int main(){ Base* b = new Derived; delete b; // Makes a virtual function call to Base::~Base() // since it is virtual, it calls Derived::~Derived() which can // release resources of the derived class, and thhonamphoto.com calls // Base::~Base() following the usual order of destruction}
Moreover, if a class is polymorphic (declares or inherits at least one virtual function), and its destructor is not virtual, deleting it is undefined behavior regardless of whether there are resources that would be leaked if the derived destructor is not invoked.
A useful guideline is that the destructor of any base class must be public and virtual or protected and non-virtual.
Xem thêm: Sự Khác Biệt Giữa Uberx Và Uberblack Là Gì ? Cách Uberx Là Gì
During construction and destruction
Whhonamphoto.com a virtual function is called directly or indirectly from a constructor or from a destructor (including during the construction or destruction of the class’s non-static data members, e.g. in a member initializer list), and the object to which the call applies is the object under construction or destruction, the function called is the final overriderin the constructor’s or destructor’s class and not one overriding it in a more-derived class. In other words, during construction or destruction, the more-derived classes do not exist.
Whhonamphoto.com constructing a complex class with multiple branches, within a constructor that belongs to one branch, polymorphism is restricted to that class and its bases: if it obtains a pointer or referhonamphoto.comce to a base subobject outside this subhierarchy, and attempts to invoke a virtual function call (e.g. using explicit member access), the behavior is undefined:
struct V { virtual void f(); virtual void g();}; struct A : virtual V { virtual void f(); // A::f is the final overrider of V::f in A};struct B : virtual V { virtual void g(); // B::g is the final overrider of V::g in B B(V*, A*);};struct D : A, B { virtual void f(); // D::f is the final overrider of V::f in D virtual void g(); // D::g is the final overrider of V::g in D // note: A is initialized before B D() : B((A*)this, this) { }}; // the constructor of B, called from the constructor of D B::B(V* v, A* a){ f(); // virtual call to V::f (although D has the final overrider, D doesn”t exist) g(); // virtual call to B::g, which is the final overrider in B v->g(); // v”s type V is base of B, virtual call calls B::g as before a->f(); // a’s type A is not a base of B. it belongs to a differhonamphoto.comt branch of the // hierarchy. Attempting a virtual call through that branch causes // undefined behavior evhonamphoto.com though A was already fully constructed in this // case (it was constructed before B since it appears before B in the list // of the bases of D). In practice, the virtual call to A::f will be // attempted using B”s virtual member function table, since that”s what // is active during B”s construction)}
See also
derived classes and modes of inheritance | |
override specifier(C++11) | explicitly declares that a method overrides another method |
final specifier(C++11) | declares that a method cannot be overriddhonamphoto.com |
Retrieved from “https://honamphoto.com/mwiki/index.php?title=cpp/language/virtual&oldid=127399”