General topics | ||||||||||||||||
Flow control | ||||||||||||||||
Conditional execution statements | ||||||||||||||||
Iteration statements (loops) | ||||||||||||||||
Jump statements | ||||||||||||||||
Functions | ||||||||||||||||
Function declaration | ||||||||||||||||
Lambda function expression | ||||||||||||||||
inline specifier | ||||||||||||||||
Dynamic exception specifications(until C++17*) | ||||||||||||||||
noexcept specifier(C++11) | ||||||||||||||||
Exceptions | ||||||||||||||||
Namespaces | ||||||||||||||||
Types | ||||||||||||||||
Specifiers | ||||||||||||||||
| ||||||||||||||||
Storage duration specifiers | ||||||||||||||||
Initialization | ||||||||||||||||
Expressions | ||||||||||||||||
Alternative representations | ||||||||||||||||
Literals | ||||||||||||||||
Boolean -Integer -Floating-point | ||||||||||||||||
Character -String -nullptr(C++11) | ||||||||||||||||
User-defined(C++11) | ||||||||||||||||
Utilities | ||||||||||||||||
Attributes(C++11) | ||||||||||||||||
Types | ||||||||||||||||
typedef declaration | ||||||||||||||||
Type alias declaration(C++11) | ||||||||||||||||
Casts | ||||||||||||||||
Memory allocation | ||||||||||||||||
Classes | ||||||||||||||||
Class-specific function properties | ||||||||||||||||
| ||||||||||||||||
Special member functions | ||||||||||||||||
| ||||||||||||||||
Templates | ||||||||||||||||
Miscellaneous | ||||||||||||||||
A copy constructor is aconstructor which can be called with an argument of the same class type and copies the content of the argument without mutating the argument.
Contents |
class-name ( parameter-list ); | (1) | ||||||||
class-name ( parameter-list ) function-body | (2) | ||||||||
class-name ( single-parameter-list ) = default; | (3) | (since C++11) | |||||||
class-name ( parameter-list ) = delete; | (4) | (since C++11) | |||||||
class-name :: class-name ( parameter-list ) function-body | (5) | ||||||||
class-name :: class-name ( single-parameter-list ) = default; | (6) | (since C++11) | |||||||
class-name | - | the class whose copy constructor is being declared |
parameter-list | - | a non-emptyparameter list satisfying all following conditions:
|
single-parameter-list | - | aparameter list of only one parameter, which is of typeT&,const T&,volatile T& orconstvolatile T& and does not have a default argument |
function-body | - | thefunction body of the copy constructor |
struct X{ X(X& other);// copy constructor// X(X other); // Error: incorrect parameter type}; union Y{ Y(Y& other,int num=1);// copy constructor with multiple parameters// Y(Y& other, int num); // Error: `num` has no default argument};
The copy constructor is called whenever an object isinitialized (bydirect-initialization orcopy-initialization) from another object of the same type (unlessoverload resolution selects a better match or the call iselided), which includes
T
;T
andf isvoid f(T t);T
, which has nomove constructor.If no user-defined copy constructors are provided for a class type, the compiler will always declare a copy constructor as a non-explicitinlinepublic member of its class. This implicitly-declared copy constructor has the formT::T(const T&) if all of the following are true:
B
ofT
has a copy constructor whose parameters are of typeconst B& orconstvolatile B&;M
ofT
of class type or array of class type has a copy constructor whose parameters are of typeconst M& orconstvolatile M&.Otherwise, the implicitly-declared copy constructor isT::T(T&).
Due to these rules, the implicitly-declared copy constructor cannot bind to avolatile lvalue argument.
A class can have multiple copy constructors, e.g. bothT::T(const T&) andT::T(T&).
Even if some user-defined copy constructors are present, the user may still force the implicit copy constructor declaration with the keyworddefault. | (since C++11) |
The implicitly-declared (or defaulted on its first declaration) copy constructor has an exception specification as described indynamic exception specification(until C++17)noexcept specification(since C++17).
If the implicitly-declared copy constructor is not deleted, it is defined (that is, a function body is generated and compiled) by the compiler ifodr-used orneeded for constant evaluation(since C++11). For union types, the implicitly-defined copy constructor copies the object representation (as bystd::memmove). For non-union class types, the constructor performs full member-wise copy of the object's direct base subobjects and member subobjects, in their initialization order, using direct initialization. For each non-static data member of a reference type, the copy constructor binds the reference to the same object or function to which the source reference is bound.
If this satisfies the requirements of aconstexpr constructor(until C++23)constexpr function(since C++23), the generated copy constructor isconstexpr. The generation of the implicitly-defined copy constructor is deprecated if | (since C++11) |
The implicitly-declared or explicitly-defaulted(since C++11) copy constructor for classT
isundefined(until C++11)defined as deleted(since C++11) if any of the following conditions is satisfied:
| (since C++11) |
T
has apotentially constructed subobject of class typeM
(or possibly multi-dimensional array thereof) such thatM
has a destructor that is deleted or(since C++11) inaccessible from the copy constructor, orM
's copy constructorThe implicitly-declared copy constructor for class | (since C++11) |
The copy constructor for classT
is trivial if all of the following are true:
T
has no virtual member functions;T
has no virtual base classes;T
is trivial;T
is trivial;A trivial copy constructor for a non-union class effectively copies every scalar subobject (including, recursively, subobject of subobjects and so forth) of the argument and performs no other action. However, padding bytes need not be copied, and even the object representations of the copied subobjects need not be the same as long as their values are identical.
TriviallyCopyable objects can be copied by copying their object representations manually, e.g. withstd::memmove. All data types compatible with the C language (POD types) are trivially copyable.
A copy constructor is eligible if it is either user-declared or both implicitly-declared and definable. | (until C++11) |
A copy constructor is eligible if it is not deleted. | (since C++11) (until C++20) |
A copy constructor is eligible if all following conditions are satisfied:
| (since C++20) |
Triviality of eligible copy constructors determines whether the class is animplicit-lifetime type, and whether the class is atrivially copyable type.
In many situations, copy constructors are optimized out even if they would produce observable side-effects, seecopy elision.
struct A{int n; A(int n=1): n(n){} A(const A& a): n(a.n){}// user-defined copy constructor}; struct B: A{// implicit default constructor B::B()// implicit copy constructor B::B(const B&)}; struct C: B{ C(): B(){}private: C(const C&);// non-copyable, C++98 style}; int main(){ A a1(7); A a2(a1);// calls the copy constructor B b; B b2= b; A a3= b;// conversion to A& and copy constructor volatile A va(10);// A a4 = va; // compile error C c;// C c2 = c; // compile error}
The following behavior-changing defect reports were applied retroactively to previously published C++ standards.
DR | Applied to | Behavior as published | Correct behavior |
---|---|---|---|
CWG 1353 | C++98 | the conditions where implicitly-declared copy constructors are undefined did not consider multi-dimensional array types | consider these types |
CWG 2094 | C++11 | volatile members make copy non-trivial (CWG issue 496) | triviality not affected |
CWG 2171 | C++11 | X(X&)=default was non-trivial | made trivial |
CWG 2595 | C++20 | a copy constructor was not eligible if there is another copy constructor which is more constrained but does not satisfy its associated constraints | it can be eligible in this case |