Movatterモバイル変換


[0]ホーム

URL:


cppreference.com
Namespaces
Variants
    Actions

      Definitions and ODR (One Definition Rule)

      From cppreference.com
      <cpp‎ |language
       
       
      C++ language
      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
      constexpr(C++11)
      consteval(C++20)
      constinit(C++20)
      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
       
       

      Definitions aredeclarations that fully define the entity introduced by the declaration. Every declaration is a definition, except for the following:

      • A function declaration without a function body:
      int f(int);// declares, but does not define f
      externconstint a;// declares, but does not define aexternconstint b=1;// defines b
      struct S{int n;// defines S::nstaticint i;// declares, but does not define S::iinlinestaticint x;// defines S::x};// defines S int S::i;// defines S::i
      • (deprecated) Namespace scope declaration of a static data member that was defined within the class with theconstexpr specifier:
      struct S{staticconstexprint x=42;// implicitly inline, defines S::x}; constexprint S::x;// declares S::x, not a redefinition
      (since C++17)
      • Declaration of a class name (byforward declaration or by the use of the elaborated type specifier in another declaration):
      struct S;// declares, but does not define S class Y f(class T p);// declares, but does not define Y and T (and also f and p)
      enum Color:int;// declares, but does not define Color
      (since C++11)
      template<typename T>// declares, but does not define T
      • A parameter declaration in a function declaration that isn't a definition:
      int f(int x);// declares, but does not define f and x int f(int x)// defines f and x{return x+ a;}
      typedef S S2;// declares, but does not define S2 (S may be incomplete)
      using S2= S;// declares, but does not define S2 (S may be incomplete)
      (since C++11)
      using N::d;// declares, but does not define d
      (since C++17)
      (since C++11)
      externtemplatef<int,char>;// declares, but does not define f<int, char>
      (since C++11)
      template<>struct A<int>;// declares, but does not define A<int>

      Anasm declaration does not define any entities, but it is classified as a definition.

      Where necessary, the compiler may implicitly define thedefault constructor,copy constructor,move constructor,copy assignment operator,move assignment operator, and thedestructor.

      If the definition of any object results in an object ofincomplete type orabstract class type, the program is ill-formed.

      Contents

      [edit]One Definition Rule

      Only one definition of any variable, function, class type, enumeration type, concept(since C++20) or template is allowed in any one translation unit (some of these may have multiple declarations, but only one definition is allowed).

      One and only one definition of every non-inline function or variable that isodr-used (see below) is required to appear in the entire program (including any standard and user-defined libraries). The compiler is not required to diagnose this violation, but the behavior of the program that violates it is undefined.

      For an inline function or inline variable(since C++17), a definition is required in every translation unit where it isodr-used .

      For a class, a definition is required wherever the class is used in a way that requires it to becomplete.

      There can be more than one definition in a program of each of the following: class type, enumeration type, inline function, inline variable(since C++17),templated entity (template or member of template, but not fulltemplate specialization), as long as all following conditions are satisfied:

      • Each definition appears in a different translation unit.
      (since C++20)
      • Each definition consists of the same sequence oftokens (typically, appears in the same header).
      • Name lookup from within each definition finds the same entities (afteroverload resolution), except that:
      • Constants with internal or no linkage may refer to different objects as long as they are not odr-used and have the same values in every definition.
      • Lambda expressions that are not in a default argument or a default template argument(since C++20) are uniquely identified by the sequence of tokens used to define them.
      (since C++11)
      • Overloaded operators, including conversion, allocation, and deallocation functions refer to the same function from each definition (unless referring to one defined within the definition).
      • Corresponding entities have the same language linkage in each definition (e.g. the include file is not inside anextern"C" block).
      • If aconst object isconstant-initialized in any of the definitions, it is constant-initialized in each definition.
      • The rules above apply to every default argument used in each definition.
      • If the definition is for a class with an implicitly-declared constructor, every translation unit where it is odr-used must call the same constructor for the base and members.
      • If the definition is for a class with a defaultedthree-way comparison, every translation unit where it is odr-used must call the same comparison operator for the base and members.
      (since C++20)
      • If the definition is for a template, then all these requirements apply to both names at the point of definition and dependent names at the point of instantiation.

      If all these requirements are satisfied, the program behaves as if there is only one definition in the entire program. Otherwise, the program is ill-formed, no diagnostic required.

      Note: in C, there is no program-wide ODR for types, and even extern declarations of the same variable in different translation units may have different typesas long as they are compatible. In C++, the source-code tokens used in declarations of the same type must be the same as described above: if one .cpp file definesstruct S{int x;}; and the other .cpp file definesstruct S{int y;};, the behavior of the program that links them together is undefined. This is usually resolved withunnamed namespaces.

      [edit]Naming an entity

      A variable isnamed by an expression if the expression is an identifier expression that denotes it.

      A function isnamed by an expression or conversion in following cases:

      • A function whose name appears as an expression or conversion (including named function, overloaded operator,user-defined conversion, user-defined placement forms ofoperator new, non-default initialization) is named by that expression if it is selected by overload resolution, except when it is an unqualified pure virtual member function or a pointer-to-member to a pure virtual function.
      • Anallocation ordeallocation function for a class is named by anew expression appearing in an expression.
      • A deallocation function for a class is named by adelete expression appearing in an expression.
      • A constructor selected to copy or move an object is considered to be named by the expression or conversion even ifcopy elision takes place.Using a prvalue in some contexts does not copy or move an object, seemandatory elision.(since C++17)

      A potentially evaluated expression or conversion odr-uses a function if it names it.

      A potentially constant evaluated expression or conversion that names a constexpr function makes itneeded for constant evaluation, which triggers definition of a defaulted function or instantiation of a function template specialization, even if the expression is unevaluated.

      (since C++11)

      [edit]Potential results

      The set ofpotential results of an expressionE is a (possibly empty) set of identifier expressions that appear withinE, combined as follows:

      • IfE is anidentifier expression, the expressionE is its only potential result.
      • IfE is a subscript expression (E1[E2]) where one of the operands is an array, the potential results of that operand is included in the set.
      • IfE is a class member access expression of the formE1.E2 orE1.template E2 naming a non-static data member, the potential results ofE1 is included in the set.
      • IfE is a class member access expression naming a static data member, the identifier expression designating the data member is included in the set.
      • IfE is a pointer-to-member access expression of the formE1.*E2 orE1.*template E2 whose second operand is a constant expression, the potential results ofE1 are included in the set.
      • IfE is an expression in parentheses ((E1)), the potential results ofE1 are included in the set.
      • IfE is a glvalue conditional expression (E1? E2: E3, whereE2 andE3 are glvalues), the union of the potential results ofE2 andE3 are both included in the set.
      • IfE is a comma expression (E1, E2), the potential results ofE2 are in the set of potential results.
      • Otherwise, the set is empty.

      [edit]ODR-use (informal definition)

      An object is odr-used if its value is read (unless it is a compile time constant) or written, its address is taken, or a reference is bound to it,

      A reference is odr-used if it is used and its referent is not known at compile time,

      A function is odr-used if a function call to it is made or its address is taken.

      If an entity is odr-used, its definition must exist somewhere in the program; a violation of that is usually a link-time error.

      struct S{staticconstint x=0;// static data member// a definition outside of class is required if it is odr-used}; constint& f(constint& r); int n= b?(1, S::x)// S::x is not odr-used here: f(S::x);// S::x is odr-used here: a definition is required

      [edit]ODR-use (formal definition)

      A variablex that is named by apotentially-evaluated expressionexpr that appears at a pointP is odr-used byexpr, unless any of the following conditions is satisfied:

      • x is a reference that isusable in constant expressions atP.
      • x is not a reference and(until C++26)expr is an element of the set of potential results of an expressionE, and any of the following conditions is satisfied:
        • E is adiscarded-value expression, and no lvalue-to-rvalue conversion is applied to it.
        • x is a non-volatile(since C++26) object that is usable in constant expressions atP and has no mutable subobjects, and any of the following conditions is satisfied:
      (since C++26)
      • E has non-volatile-qualified non-class type, and the lvalue-to-rvalue conversion is applied to it.
      struct S{staticconstint x=1;};// applying lvalue-to-rvalue conversion// to S::x yields a constant expression int f(){    S::x;// discarded-value expression does not odr-use S::x return S::x;// expression where lvalue-to-rvalue conversion// applies does not odr-use S::x}

      *this is odr-used ifthis appears as a potentially-evaluated expression (including the implicitthis in a non-static member function call expression).

      Astructured binding is odr-used if it appears as a potentially-evaluated expression.

      (since C++17)

      A function is odr-used in following cases:

      • A function is odr-used if it is named by (see below) a potentially-evaluated expression or conversion.
      • Avirtual member function is odr-used if it is not a pure virtual member function (addresses of virtual member functions are required to construct the vtable).
      • A non-placement allocation or deallocation function for a class is odr-used by the definition of a constructor of that class.
      • A non-placement deallocation function for a class is odr-used by the definition of the destructor of that class, or by being selected by the lookup at the point of definition of a virtual destructor.
      • An assignment operator in a classT that is a member or base of another classU is odr-used by an implicitly-defined copy-assignment or move-assignment functions ofU.
      • A constructor (including default constructors) for a class is odr-used by theinitialization that selects it.
      • A destructor for a class is odr-used if it ispotentially invoked.
      This section is incomplete
      Reason: list of all situations where odr-use makes a difference

      [edit]Defect reports

      The following behavior-changing defect reports were applied retroactively to previously published C++ standards.

      DRApplied toBehavior as publishedCorrect behavior
      CWG 261C++98a deallocation function for a polymorphic class
      might be odr-used even if there were no relevant
      new or delete expressions in the program
      supplemented the
      odr-use cases to cover
      constructors and destructors
      CWG 678C++98an entity could have definitions
      with different language linkages
      the behavior is
      undefined in this case
      CWG 1472C++98reference variables which satisfy the requirements for
      appearing in a constant expression were odr-used even
      if the lvalue-to-rvalue conversion is immediately applied
      they are not
      odr-used in this case
      CWG 1614C++98taking address of a pure virtual function odr-used itthe function is not odr-used
      CWG 1741C++98constant objects that are immediately lvalue-to-rvalue
      converted in potentially-evaluated expressions were odr-used
      they are not odr-used
      CWG 1926C++98array subscript expressions did not propagate potential resultsthey propagate
      CWG 2242C++98it was unclear whether aconst object that is only
      constant-initialized in part of its definitions violates ODR
      ODR is not violated; the object is
      constant-initialized in this case
      CWG 2300C++11lambda expressions in different translation
      units could never have the same closure type
      the closure type can be the
      same under one definition rule
      CWG 2353C++98a static data member was not a potential result
      of a member access expression accessing it
      it is
      CWG 2433C++14a variable template could not have
      multiple definitions in a program
      it can

      [edit]References

      • C++23 standard (ISO/IEC 14882:2024):
      • 6.3 One definition rule [basic.def.odr]
      • C++20 standard (ISO/IEC 14882:2020):
      • 6.3 One definition rule [basic.def.odr]
      • C++17 standard (ISO/IEC 14882:2017):
      • 6.2 One definition rule [basic.def.odr]
      • C++14 standard (ISO/IEC 14882:2014):
      • 3.2 One definition rule [basic.def.odr]
      • C++11 standard (ISO/IEC 14882:2011):
      • 3.2 One definition rule [basic.def.odr]
      • C++03 standard (ISO/IEC 14882:2003):
      • 3.2 One definition rule [basic.def.odr]
      • C++98 standard (ISO/IEC 14882:1998):
      • 3.2 One definition rule [basic.def.odr]
      Retrieved from "https://en.cppreference.com/mwiki/index.php?title=cpp/language/definition&oldid=178487"

      [8]ページ先頭

      ©2009-2025 Movatter.jp