Namespaces
Variants
Views
Actions

Difference between revisions of "cpp/language/storage duration"

From cppreference.com
< cpp‎ | language
(Added P2788R0 DR.)
m (Internal linkage: ~{rlpsd})
 
(27 intermediate revisions by 7 users not shown)
Line 2: Line 2:
 
{{cpp/language/declarations/navbar}}
 
{{cpp/language/declarations/navbar}}
  
The storage class specifiers are a part of the {{spar|decl-specifier-seq}} of a name's {{rlp|declarations|declaration syntax}}. Together with the {{rlp|scope}} of the name, they control two independent properties of the name: its ''storage duration'' and its ''linkage''.  
+
The storage class specifiers are a part of the {{spar|decl-specifier-seq}} of a name's {{rlp|declarations|declaration syntax}}. Together with the {{rlp|scope}} of the name, they control two independent properties of the name: its ''storage duration'' and its ''linkage''.
  
{{rev begin}}
+
===Storage duration===
:*no specifier {{rev inl|until=c++11| or {{ttb|auto}}}} - ''automatic'' storage duration.
+
The ''storage duration'' is the property of an {{rlp|object}} that defines the minimum potential lifetime of the storage containing the object. The storage duration is determined by the construct used to create the object and is one of the following:
{{rev|until=c++17|
+
* static storage duration
:*{{ttb|register}} - ''automatic'' storage duration. Also hints to the compiler to place the object in the processor's register. {{mark deprecated}}
+
}}
+
{{rev end}}
+
:*{{ttb|static}} - ''static'' or ''thread'' storage duration and ''internal'' linkage (or ''external'' linkage for static class members not in an anonymous namespace).
+
:*{{ttb|extern}} - ''static'' or ''thread'' storage duration and ''external'' linkage.
+
 
{{rrev|since=c++11|
 
{{rrev|since=c++11|
:*{{ttb|thread_local}} - ''thread'' storage duration.
+
* thread storage duration
 
}}
 
}}
:*{{ttb|mutable}} - does not affect storage duration or linkage. See {{rlp|cv|const/volatile}} for the explanation.
+
* automatic storage duration
 +
* dynamic storage duration
  
 +
Static{{rev inl|since=c++11|, thread,}} and automatic storage durations are associated with objects introduced by {{rlp|declarations}} and with {{rlp|lifetime#Temporary object lifetime|temporary objects}}. The dynamic storage duration is associated with objects created by a {{rlp|new|{{c/core|new}} expression}} or with {{rlp|object#Object creation|implicitly created objects}}.
  
Only one storage class specifier may appear in a declaration {{rev inl|since=c++11|except that {{tt|thread_local}} may be combined with {{tt|static}} or with {{tt|extern}}}}.
+
The storage duration categories apply to references as well.
  
===Explanation===
+
The storage duration of {{rlpsd|object#Subobjects}} and reference members is that of their complete object.
 +
 
 +
====Specifiers====
 +
The following keywords are ''storage class specifiers''{{sep}}:
 
{{rev begin}}
 
{{rev begin}}
 
{{rev|until=c++11|
 
{{rev|until=c++11|
@1@ The {{tt|auto}} specifier was only allowed for objects declared at block scope or in function parameter lists. It indicated automatic storage duration, which is the default for these kinds of declarations. The meaning of this keyword was changed in C++11.
+
* {{c/core|auto}}
 
}}
 
}}
 
{{rev|until=c++17|
 
{{rev|until=c++17|
@2@ The {{tt|register}} specifier is only allowed for objects declared at block scope and in function parameter lists. It indicates automatic storage duration, which is the default for these kinds of declarations. Additionally, the presence of this keyword may be used as a hint for the optimizer to store the value of this variable in a CPU register. This keyword is deprecated.
+
* {{c/core|register}}
 
}}
 
}}
 
{{rev end}}
 
{{rev end}}
@3@ The {{tt|static}} specifier is only allowed in the declarations of objects (except in function parameter lists), declarations of functions (except at block scope), and declarations of anonymous unions. When used in a declaration of a class member, it declares a {{rlp|static|static member}}. When used in a declaration of an object, it specifies static storage duration (except if accompanied by {{tt|thread_local}}). When used in a declaration at namespace scope, it specifies internal linkage.
+
* {{c/core|static}}
@4@ The {{tt|extern}} specifier is only allowed in the declarations of variables and functions (except class members or function parameters). It specifies external linkage, and does not technically affect storage duration, but it cannot be used in a definition of an automatic storage duration object, so all {{tt|extern}} objects have static or thread durations. In addition, a variable declaration that uses {{tt|extern}} and has no initializer is not a {{rlp|definition}}.
+
 
+
 
{{rrev|since=c++11|
 
{{rrev|since=c++11|
@5@ The {{tt|thread_local}} keyword is only allowed for objects declared at namespace scope, objects declared at block scope, and static data members. It indicates that the object has thread storage duration. If {{tt|thread_local}} is the only storage class specifier applied to a block scope variable, {{tt|static}} is also implied. It can be combined with {{tt|static}} or {{tt|extern}} to specify internal or external linkage (except for static data members which always have external linkage) respectively.
+
* {{c/core|thread_local}}
 
}}
 
}}
 +
* {{c/core|extern}}
 +
* {{c/core|mutable}}
  
====Storage duration====
+
In a {{spar|decl-specifier-seq}}, there can be at most one storage class specifier{{rev inl|since=c++11|, except that {{c/core|thread_local}} may appear with {{c/core|static}} or {{c/core|extern}}}}.
All {{rlp|object}}s in a program have one of the following storage durations:
+
  
:*'''''automatic''''' storage duration. The storage for the object is allocated at the beginning of the enclosing code block and deallocated at the end. All local objects have this storage duration, except those declared {{tt|static}}, {{tt|extern}} or {{tt|thread_local}}.
+
{{c/core|mutable}} has no effect on storage duration. For its usage, see {{rlp|cv|const/volatile}}.
  
:*'''''static''''' storage duration. The storage for the object is allocated when the program begins and deallocated when the program ends. Only one instance of the object exists. All objects declared at namespace scope (including global namespace) have this storage duration, plus those declared with {{tt|static}} or {{tt|extern}}. See {{rlp|initialization#Non-local variables|Non-local variables}} and [[#Static local variables|Static local variables]] for details on initialization of objects with this storage duration.
+
Other storage class specifiers can appear in the {{spar sep|decl-specifier-seq}}s of the following declarations:
 +
{|class="wikitable" style="text-align: center;"
 +
!rowspan=4|Specifier
 +
!colspan=9|Can appear in the {{spar sep|decl-specifier-seq}}s of
 +
|-
 +
!colspan=4|Variable declarations
 +
!colspan=3|Function declarations
 +
!rowspan=3|Structured binding declarations<br>{{mark since c++17}}
 +
|-
 +
!colspan=2|Non-member
 +
!colspan=2|Member
 +
!rowspan=2|Non-member
 +
!colspan=2|Member
 +
|-
 +
!Non-parameter
 +
!Function parameter
 +
!Non-static
 +
!{{nbsp}}Static{{nbsp}}
 +
!Non-static
 +
!{{nbsp}}Static{{nbsp}}
 +
|-
 +
|{{c/core|auto}}
 +
|{{maybe|Block scope only}}||{{yes}}||{{no}}||{{no}}
 +
|{{no}}||{{no}}||{{no}}||{{n/a}}
 +
|-
 +
|{{c/core|register}}
 +
|{{maybe|Block scope only}}||{{yes}}||{{no}}||{{no}}
 +
|{{no}}||{{no}}||{{no}}||{{n/a}}
 +
|-
 +
|{{c/core|static}}
 +
|{{yes}}||{{no}}||colspan=2 {{yes|Declares static}}
 +
|{{maybe|Namespace scope only}}||colspan=2 {{yes|Declares static}}||{{yes}}
 +
|-
 +
|{{nbsp}}{{c/core|thread_local}}{{nbsp}}
 +
|{{yes}}||{{no}}||{{no}}||{{yes}}
 +
|{{no}}||{{no}}||{{no}}||{{yes}}
 +
|-
 +
|{{c/core|extern}}
 +
|{{yes}}||{{no}}||{{no}}||{{no}}
 +
|{{yes}}||{{no}}||{{no}}||{{no}}
 +
|}
  
 +
{{rlps|union#Anonymous unions}} can also be declared with {{c/core|static}}.
 +
 +
{{rrev|until=c++17|
 +
{{c/core|register}} is a hint that the variable so declared will be heavily used, so that its value can be stored in a CPU register. The hint can be ignored, and in most implementations it will be ignored if the address of the variable is taken. This use is deprecated.
 +
}}
 +
 +
====Static storage duration====
 +
A variable satisfying all following conditions has ''static storage duration''{{sep}}:
 +
* It belongs to a {{rlpsd|scope#Namespace scope}} or are first declared with {{c/core|static}} or {{c/core|extern}}.
 
{{rrev|since=c++11|
 
{{rrev|since=c++11|
:*'''''thread''''' storage duration. The storage for the object is allocated when the thread begins and deallocated when the thread ends. Each thread has its own instance of the object. Only objects declared {{tt|thread_local}} have this storage duration. {{tt|thread_local}} can appear together with {{tt|static}} or {{tt|extern}} to adjust linkage. See {{rlp|initialization#Non-local variables|Non-local variables}} and [[#Static local variables|Static local variables]] for details on initialization of objects with this storage duration.
+
* It does not have thread storage duration.
 
}}
 
}}
  
:*'''''dynamic''''' storage duration. The storage for the object is allocated and deallocated upon request by using [[cpp/memory|dynamic memory allocation]] functions. See {{rlp|new|new-expression}} for details on initialization of objects with this storage duration.
+
The storage for these entities lasts for the duration of the program.
  
The storage duration of {{rlp|object#Subobjects|subobjects}} and reference members is that of their complete object.
+
{{rrev|since=c++11|
 +
====Thread storage duration====
 +
All variables declared with {{c/core|thread_local}} have ''thread storage duration''.
  
====Linkage====
+
The storage for these entities lasts for the duration of the thread in which they are created. There is a distinct object or reference per thread, and use of the declared name refers to the entity associated with the current thread.
A name that denotes object, reference, function, type, template, namespace, or value, may have ''linkage''. If a name has linkage, it refers to the same entity as the same name introduced by a declaration in another scope. If a variable, function, or another entity with the same name is declared in several scopes, but does not have sufficient linkage, then several instances of the entity are generated.
+
}}
  
The following linkages are recognized:
+
====Automatic storage duration====
 +
The following variables have ''automatic storage duration''{{sep}}:
 +
* Variables that belong to a {{rlpsd|scope#Block scope}} and are not explicitly declared {{c/core|static}}{{rev inl|since=c++11|, {{c/core|thread_local}},}} or {{c/core|extern}}<!-- no need to mention “auto” and “register” here -->. The storage for such variables lasts until the block in which they are created exits.
 +
* Variables that belong to a parameter scope (i.e. function parameters). The storage for a function parameter lasts until immediately after its {{rlp|operator other#Built-in function call operator|destruction}}.
  
=====no linkage=====
+
====Dynamic storage duration====
The name can be referred to only from the scope it is in.
+
Objects created by the following methods during program execution have ''dynamic storage duration''{{sep}}:
 +
* {{rlp|new|{{c/core|new}} expressions}}. The storage for such objects is allocated by [[cpp/memory/new/operator new|allocation functions]] and deallocated by [[cpp/memory/new/operator delete|deallocation functions]].
 +
* {{rlp|object#Object creation|Implicitly creation}} by other means. The storage for such objects overlaps with some existing storage.
 +
* {{rlps|throw#Exception object}}s. The storage for such objects is allocated and deallocated in an unspecified way.
  
 +
===Linkage===
 +
A name can have ''external linkage''{{sep}}{{rev inl|since=c++20|, ''module linkage''}}, ''internal linkage'', or ''no linkage'':
 +
* An entity whose name has external linkage can be {{rlp|conflicting declarations#Multiple declarations of the same entity|redeclared}} in another {{rlp|translation phases|translation unit}}{{rev inl|since=c++20|, and the redeclaration can be {{rlp|modules#Module ownership|attached to a different module}}}}.
 +
{{rrev|since=c++20|
 +
* An entity whose name has module linkage can be redeclared in another translation unit, as long as the redeclaration is attached to the same module.
 +
}}
 +
* An entity whose name has internal linkage can be redeclared in another scope in the same translation unit.
 +
* An entity whose name has no linkage can only be redeclared in the same scope.
 +
 +
The following linkages are recognized:
 +
 +
====No linkage====
 
Any of the following names declared at block scope have no linkage:
 
Any of the following names declared at block scope have no linkage:
* variables that aren't explicitly declared {{tt|extern}} (regardless of the {{tt|static}} modifier);
+
* variables that are not explicitly declared {{c/core|extern}} (regardless of the {{c/core|static}} modifier);
* {{rlp|class#Local classes|local classes}} and their member functions;
+
* {{rlpsd|class#Local classes}} and their member functions;
 
* other names declared at block scope such as typedefs, enumerations, and enumerators.
 
* other names declared at block scope such as typedefs, enumerations, and enumerators.
  
 
Names not specified with external{{rev inl|since=c++20|, module,}} or internal linkage also have no linkage, regardless of which scope they are declared in.
 
Names not specified with external{{rev inl|since=c++20|, module,}} or internal linkage also have no linkage, regardless of which scope they are declared in.
  
=====internal linkage=====
+
====Internal linkage====
The name can be referred to from all scopes in the current translation unit.
+
 
+
 
Any of the following names declared at namespace scope have internal linkage:
 
Any of the following names declared at namespace scope have internal linkage:
* variables{{rev inl|since=c++14|, variable templates}}, functions, or function templates declared {{tt|static}};
+
* variables{{rev inl|since=c++14|, variable templates}}, functions, or function templates declared {{c/core|static}};
 
* {{rev inl|since=c++14|non-template }}variables of non-volatile const-qualified type, unless
 
* {{rev inl|since=c++14|non-template }}variables of non-volatile const-qualified type, unless
{{rrev|since=c++17|
+
{{rev begin}}
 +
{{rev|since=c++17|
 
:* they are inline,
 
:* they are inline,
 
}}
 
}}
{{rrev|since=c++20|
+
{{rev|since=c++20|
:* they are declared in the purview of a {{rlp|modules#Module declarations|module interface unit}} (outside the {{rlp|modules#Private module fragment|private module fragment}}, if any) or {{rlp|modules#Module partitions|module partition}},
+
:* they are declared in the purview of a {{rlp|modules#Module declarations|module interface unit}} (outside the {{rlpsd|modules#Private module fragment}}, if any) or {{rlp|modules#Module partitions|module partition}},
 
}}
 
}}
 +
{{rev end}}
 
:* they are explicitly declared {{c/core|extern}}, or
 
:* they are explicitly declared {{c/core|extern}}, or
 
:* they were previously declared and the prior declaration did not have internal linkage;
 
:* they were previously declared and the prior declaration did not have internal linkage;
Line 83: Line 151:
  
 
{{rrev|since=c++11|
 
{{rrev|since=c++11|
In addition, all names declared in {{rlp|namespace|unnamed namespace}} or a namespace within an unnamed namespace, even ones explicitly declared {{tt|extern}}, have internal linkage.
+
In addition, all names declared in {{rlpsd|namespace#Unnamed namespaces}} or a namespace within an unnamed namespace, even ones explicitly declared {{c/core|extern}}, have internal linkage.
 
}}
 
}}
  
=====external linkage=====
+
====External linkage====
The name can be referred to from the scopes in the other translation units. Variables and functions with external linkage also have {{rlp|language linkage}}, which makes it possible to link translation units written in different programming languages.
+
Variables and functions with external linkage also have {{rlp|language linkage}}, which makes it possible to link translation units written in different programming languages.
  
 
Any of the following names declared at namespace scope have external linkage, unless they are declared in an unnamed namespace{{rev inl|since=c++20| or their declarations are attached to a named module and are not exported}}:
 
Any of the following names declared at namespace scope have external linkage, unless they are declared in an unnamed namespace{{rev inl|since=c++20| or their declarations are attached to a named module and are not exported}}:
* variables and functions not listed above (that is, functions not declared {{tt|static}}, non-const variables not declared {{tt|static}}, and any variables declared {{tt|extern}});
+
* variables and functions not listed above (that is, functions not declared {{c/core|static}}, non-const variables not declared {{c/core|static}}, and any variables declared {{c/core|extern}});
 
* enumerations;
 
* enumerations;
 
* names of classes, their member functions, static data members (const or not), nested classes and enumerations, and functions first introduced with {{rlp|friend}} declarations inside class bodies;
 
* names of classes, their member functions, static data members (const or not), nested classes and enumerations, and functions first introduced with {{rlp|friend}} declarations inside class bodies;
* names of all templates not listed above (that is, not function templates declared {{tt|static}}).
+
* names of all templates not listed above (that is, not function templates declared {{c/core|static}}).
  
 
Any of the following names first declared at block scope have external linkage:
 
Any of the following names first declared at block scope have external linkage:
* names of variables declared {{tt|extern}};
+
* names of variables declared {{c/core|extern}};
 
* names of functions.
 
* names of functions.
  
 
{{rrev|since=c++20|
 
{{rrev|since=c++20|
=====module linkage=====
+
====Module linkage====
The name can be referred to only from the scopes in the same module unit or in the other translation units of the same named module.
+
Names declared at namespace scope have module linkage if their declarations are attached to a named module and are not exported, and do not have internal linkage.
 
+
Names declared at namespace scope have module linkage if their declarations are attached to a named module and are not exported, and don't have internal linkage.
+
 
}}
 
}}
  
 
{{todo|add the description of the behavior when an entity is declared with different linkages in the same translation unit (6.6 paragraph 6), note the difference between C++20 (ill-formed) and the current draft (well-formed)}}
 
{{todo|add the description of the behavior when an entity is declared with different linkages in the same translation unit (6.6 paragraph 6), note the difference between C++20 (ill-formed) and the current draft (well-formed)}}
  
===Static local variables===
+
===Static block variables===
Variables declared at block scope with the specifier {{tt|static}} {{rev inl|since=c++11|or {{tt|thread_local}}}} have static {{rev inl|since=c++11|or thread}} storage duration but are initialized the first time control passes through their declaration (unless their initialization is {{rlp|zero initialization|zero-}} or {{rlp|constant initialization|constant-initialization}}, which can be performed before the block is first entered). On all further calls, the declaration is skipped.
+
Block variables with static{{rev inl|since=c++11| or thread}} storage duration are initialized the first time control passes through their declaration (unless their initialization is {{rlp|zero initialization|zero-}} or {{rlp|constant initialization|constant-initialization}}, which can be performed before the block is first entered). On all further calls, the declaration is skipped.
 
+
* If the initialization {{rlp|throw|throws an exception}}, the variable is not considered to be initialized, and initialization will be attempted again the next time control passes through the declaration.
If the initialization {{rlp|throw|throws an exception}}, the variable is not considered to be initialized, and initialization will be attempted again the next time control passes through the declaration.
+
* If the initialization recursively enters the block in which the variable is being initialized, the behavior is undefined.
 
+
If the initialization recursively enters the block in which the variable is being initialized, the behavior is undefined.
+
 
+
 
{{rrev|since=c++11|
 
{{rrev|since=c++11|
If multiple threads attempt to initialize the same static local variable concurrently, the initialization occurs exactly once (similar behavior can be obtained for arbitrary functions with {{lc|std::call_once}}).
+
* If multiple threads attempt to initialize the same static local variable concurrently, the initialization occurs exactly once (similar behavior can be obtained for arbitrary functions with {{lc|std::call_once}}).
 
+
:* Usual implementations of this feature use variants of the double-checked locking pattern, which reduces runtime overhead for already-initialized local statics to a single non-atomic boolean comparison.
Note: usual implementations of this feature use variants of the double-checked locking pattern, which reduces runtime overhead for already-initialized local statics to a single non-atomic boolean comparison.
+
 
}}
 
}}
  
The destructor for a block-scope static variable [[cpp/utility/program/exit|is called at program exit]], but only if the initialization took place successfully.
+
The destructor for a block variable with static storage duration [[cpp/utility/program/exit|is called at program exit]], but only if the initialization took place successfully.
  
Function-local static objects in all definitions of the same {{rlp|inline|inline function}} (which may be implicitly inline) all refer to the same object defined in one translation unit, as long as the function has external linkage.
+
Variables with static storage duration in all definitions of the same {{rlp|inline|inline function}} (which may be implicitly inline) all refer to the same object defined in one translation unit, as long as the function has external linkage.
  
 
===Translation-unit-local entities===
 
===Translation-unit-local entities===
Line 139: Line 201:
 
* initializer for a variable or variable template
 
* initializer for a variable or variable template
 
* friend declarations in a class definition
 
* friend declarations in a class definition
* use of value of a variable, if the variable is {{rlp|constant expression#Usable in constant expressions|usable in constant expressions}}
+
* use of value of a variable, if the variable is {{rlpsd|constant expression#Usable in constant expressions}}
  
 
{{rrev|since=c++20|
 
{{rrev|since=c++20|
Line 148: Line 210:
  
 
===Notes===
 
===Notes===
Names at the top-level namespace scope (file scope in C) that are {{tt|const}} and not {{tt|extern}} have external linkage in C, but internal linkage in C++.
+
Names at the top-level namespace scope (file scope in C) that are {{c/core|const}} and not {{c/core|extern}} have external linkage in C, but internal linkage in C++.
  
Since C++11, {{tt|auto}} is no longer a storage class specifier; it is used to indicate type deduction.  
+
Since C++11, {{c/core|auto}} is no longer a storage class specifier; it is used to indicate type deduction.  
  
 
{{rrev multi|until1=c++17
 
{{rrev multi|until1=c++17
 
|rev1=
 
|rev1=
In C, the address of a {{tt|register}} variable cannot be taken, but in C++, a variable declared {{tt|register}} is semantically indistinguishable from a variable declared without any storage class specifiers.
+
In C, the address of a {{c/core|register}} variable cannot be taken, but in C++, a variable declared {{c/core|register}} is semantically indistinguishable from a variable declared without any storage class specifiers.
 
|rev2=
 
|rev2=
In C++, unlike C, variables cannot be declared {{tt|register}}.
+
In C++, unlike C, variables cannot be declared {{c/core|register}}.
 
}}
 
}}
  
Names of {{tt|thread_local}} variables with internal or external linkage referred from different scopes may refer to the same or to different instances depending on whether the code is executing in the same or in different threads.
+
Names of {{c/core|thread_local}} variables with internal or external linkage referred from different scopes may refer to the same or to different instances depending on whether the code is executing in the same or in different threads.
  
The {{tt|extern}} keyword can also be used to specify {{rlp|language linkage}} and {{rlp|class template|explicit template instantiation declarations}}, but it's not a storage class specifier in those cases (except when a declaration is directly contained in a language linkage specification, in which case the declaration is treated as if it contains the {{tt|extern}} specifier).
+
The {{c/core|extern}} keyword can also be used to specify {{rlp|language linkage}} and {{rlp|class template|explicit template instantiation declarations}}, but it's not a storage class specifier in those cases (except when a declaration is directly contained in a language linkage specification, in which case the declaration is treated as if it contains the {{c/core|extern}} specifier).
  
{{rlp|cv|The keyword {{tt|mutable}}}} is a storage class specifier in the C++ language grammar, although it doesn't affect storage duration or linkage.
+
Storage class specifiers, except for {{c/core|thread_local}}, are not allowed on {{rlp|template specialization|explicit specializations}} and {{rlpsd|class template#Explicit instantiation}}s:
 
+
{{todo|the rules about re-declaring names in the same TU}}
+
 
+
Storage class specifiers, except for {{tt|thread_local}}, are not allowed on {{rlp|template specialization|explicit specializations}} and {{rlp|class template#Explicit instantiation|explicit instantiations}}:
+
 
{{source|1=
 
{{source|1=
 
template<class T>
 
template<class T>
Line 178: Line 236:
 
thread_local int S<float>::tlm = 0; // "static" does not appear here
 
thread_local int S<float>::tlm = 0; // "static" does not appear here
 
}}
 
}}
{{feature test macro|__cpp_threadsafe_static_init|[[#Static_local_variables|Dynamic initialization and destruction with concurrency]]|value=200806L|std=C++11}}
+
 
 +
{{rev begin}}
 +
{{rev|since=c++14|
 +
A {{c/core|const}} (may be implied by {{c/core|constexpr}}) variable template used to have internal linkage by default, which was inconsistent with other templated entities. Defect report {{wg21|CWG2387}} corrected this.
 +
}}
 +
{{rev|since=c++17|1=
 +
{{rlpt|inline}} acts as a workaround for {{wg21|CWG2387}} by giving external linkage by default. This is why the {{c/core|inline}} was [https://wg21.link/p0607r0 added] to many variable templates and then [https://github.com/cplusplus/draft/pull/4625 removed] after having CWG2387 accepted. Standard library implementations also need to use {{c/core|inline}} as long as a supported compiler has not get CWG2387 implemented. See [https://gcc.gnu.org/bugzilla/show_bug.cgi?id=109126 GCC Bugzilla #109126] and [https://github.com/microsoft/STL/pull/4546 MSVC STL PR #4546].
 +
}}
 +
{{rev end}}
 +
 
 +
{{feature test macro|__cpp_threadsafe_static_init|[[#Static local variables|Dynamic initialization and destruction with concurrency]]|value=200806L|std=C++11}}
  
 
===Keywords===
 
===Keywords===
Line 190: Line 258:
 
===Example===
 
===Example===
 
{{example
 
{{example
|p=true
 
 
|code=
 
|code=
 
#include <iostream>
 
#include <iostream>
 +
#include <mutex>
 
#include <string>
 
#include <string>
 
#include <thread>
 
#include <thread>
#include <mutex>
 
  
thread_local unsigned int rage = 1;  
+
thread_local unsigned int rage = 1;
 
std::mutex cout_mutex;
 
std::mutex cout_mutex;
  
Line 210: Line 277:
 
{
 
{
 
     std::thread a(increase_rage, "a"), b(increase_rage, "b");
 
     std::thread a(increase_rage, "a"), b(increase_rage, "b");
 
+
   
 
     {
 
     {
 
         std::lock_guard<std::mutex> lock(cout_mutex);
 
         std::lock_guard<std::mutex> lock(cout_mutex);
 
         std::cout << "Rage counter for main: " << rage << '\n';
 
         std::cout << "Rage counter for main: " << rage << '\n';
 
     }
 
     }
 
+
   
 
     a.join();
 
     a.join();
 
     b.join();
 
     b.join();
 
}
 
}
 +
|p=true
 
|output=
 
|output=
 
Rage counter for a: 2
 
Rage counter for a: 2
Line 231: Line 299:
 
{{dr list item|wg=cwg|dr=426|std=C++98|before=an entity could be declared with both internal<br>and external linkage in the same translation unit|after=the program is ill-formed in this case}}
 
{{dr list item|wg=cwg|dr=426|std=C++98|before=an entity could be declared with both internal<br>and external linkage in the same translation unit|after=the program is ill-formed in this case}}
 
{{dr list item|wg=cwg|dr=527|std=C++98|before=the type restriction introduced by the resolution of CWG<br>389 was also applied to variables and functions that<br>cannot be named outside their own translation units|after=the restriction is lifted for these<br>variables and functions (i.e. with no<br>linkage or internal linkage, or declared<br>within unnamed namespaces)}}
 
{{dr list item|wg=cwg|dr=527|std=C++98|before=the type restriction introduced by the resolution of CWG<br>389 was also applied to variables and functions that<br>cannot be named outside their own translation units|after=the restriction is lifted for these<br>variables and functions (i.e. with no<br>linkage or internal linkage, or declared<br>within unnamed namespaces)}}
{{dr list item|wg=cwg|dr=809|std=C++98|before={{c|register}} served very little function|after=deprecated}}
+
{{dr list item|wg=cwg|dr=809|std=C++98|before={{c/core|register}} served very little function|after=deprecated}}
{{dr list item|wg=cwg|dr=1648|std=C++11|before={{tt|static}} was implied even if<br>{{tt|thread_local}} is combined with {{tt|extern}}|after=implied only if no other storage<br>class specifier is present}}
+
{{dr list item|wg=cwg|dr=1648|std=C++11|before={{c/core|static}} was implied even if<br>{{c/core|thread_local}} is combined with {{c/core|extern}}|after=implied only if no other storage<br>class specifier is present}}
{{dr list item|wg=cwg|dr=1686|std=C++98<br>C++11|before=the name of a non-static variable declared in namespace<br>scope had internal linkage only if it is explicitly<br>declared {{c|const}} (C++98) or {{c|constexpr}} (C++11)|after=only required the type<br>to be const-qualified}}
+
{{dr list item|wg=cwg|dr=1686|std=C++98<br>C++11|before=the name of a non-static variable declared in namespace<br>scope had internal linkage only if it is explicitly<br>declared {{c/core|const}} (C++98) or {{c/core|constexpr}} (C++11)|after=only required the type<br>to be const-qualified}}
 
{{dr list item|wg=cwg|dr=2019|std=C++98|before=the storage duration of reference<br>members were unspecified|after=same as their complete object}}
 
{{dr list item|wg=cwg|dr=2019|std=C++98|before=the storage duration of reference<br>members were unspecified|after=same as their complete object}}
 
{{dr list item|wg=cwg|dr=2387|std=C++14|before=unclear whether const-qualified variable<br>template have internal linkage by default|after=const qualifier does not affect<br>the linkage of variable<br>templates or their instances}}
 
{{dr list item|wg=cwg|dr=2387|std=C++14|before=unclear whether const-qualified variable<br>template have internal linkage by default|after=const qualifier does not affect<br>the linkage of variable<br>templates or their instances}}
 +
{{dr list item|wg=cwg|dr=2533|std=C++98|before=the storage duration of implicitly-<br>created objects were unclear|after=made clear}}
 +
{{dr list item|wg=cwg|dr=2850|std=C++98|before=it was unclear when the storage for<br>function parameters are deallocated|after=made clear}}
 +
{{dr list item|wg=cwg|dr=2872|std=C++98|before=the meaning of “can be referred to” was unclear|after=improved wording}}
 
{{dr list item|paper=P2788R0|std=C++20|before=declaring a const-qualified variable in a namespace<br>gave it internal linkage even in a module unit|after=internal linkage is not given}}
 
{{dr list item|paper=P2788R0|std=C++20|before=declaring a const-qualified variable in a namespace<br>gave it internal linkage even in a module unit|after=internal linkage is not given}}
 
{{dr list end}}
 
{{dr list end}}
 +
 +
===References===
 +
{{ref std c++23}}
 +
{{ref std|section=6.7.5|title=Storage duration|id=basic.stc}}
 +
{{ref std end}}
 +
{{ref std c++20}}
 +
{{ref std|section=6.7.5|title=Storage duration|id=basic.stc}}
 +
{{ref std end}}
 +
{{ref std c++17}}
 +
{{ref std|section=6.7|title=Storage duration|id=basic.stc}}
 +
{{ref std end}}
 +
{{ref std c++14}}
 +
{{ref std|section=3.7|title=Storage duration|id=basic.stc}}
 +
{{ref std end}}
 +
{{ref std c++11}}
 +
{{ref std|section=3.7|title=Storage duration|id=basic.stc}}
 +
{{ref std end}}
 +
{{ref std c++03}}
 +
{{ref std|section=3.7|title=Storage duration|id=basic.stc}}
 +
{{ref std end}}
 +
{{ref std c++98}}
 +
{{ref std|section=3.7|title=Storage duration|id=basic.stc}}
 +
{{ref std end}}
  
 
===See also===
 
===See also===

Latest revision as of 19:44, 1 November 2024

 
 
C++ language
General topics
Flow control
Conditional execution statements
if
Iteration statements (loops)
for
range-for (C++11)
Jump statements
Functions
Function declaration
Lambda function expression
inline specifier
Dynamic exception specifications (until C++17*)
noexcept specifier (C++11)
Exceptions
Namespaces
Types
Specifiers
const/volatile
decltype (C++11)
auto (C++11)
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
explicit (C++11)
static

Special member functions
Templates
Miscellaneous
 
 

The storage class specifiers are a part of the decl-specifier-seq of a name's declaration syntax. Together with the scope of the name, they control two independent properties of the name: its storage duration and its linkage.

Contents

[edit] Storage duration

The storage duration is the property of an object that defines the minimum potential lifetime of the storage containing the object. The storage duration is determined by the construct used to create the object and is one of the following:

  • static storage duration
  • thread storage duration
(since C++11)
  • automatic storage duration
  • dynamic storage duration

Static, thread,(since C++11) and automatic storage durations are associated with objects introduced by declarations and with temporary objects. The dynamic storage duration is associated with objects created by a new expression or with implicitly created objects.

The storage duration categories apply to references as well.

The storage duration of subobjects and reference members is that of their complete object.

[edit] Specifiers

The following keywords are storage class specifiers :

  • auto
(until C++11)
  • register
(until C++17)
  • static
  • thread_local
(since C++11)
  • extern
  • mutable

In a decl-specifier-seq, there can be at most one storage class specifier, except that thread_local may appear with static or extern(since C++11).

mutable has no effect on storage duration. For its usage, see const/volatile.

Other storage class specifiers can appear in the decl-specifier-seq s of the following declarations:

Specifier Can appear in the decl-specifier-seq s of
Variable declarations Function declarations Structured binding declarations
(since C++17)
Non-member Member Non-member Member
Non-parameter Function parameter Non-static  Static  Non-static  Static 
auto Block scope only Yes No No No No No N/A
register Block scope only Yes No No No No No N/A
static Yes No Declares static Namespace scope only Declares static Yes
 thread_local  Yes No No Yes No No No Yes
extern Yes No No No Yes No No No

Anonymous unions can also be declared with static.

register is a hint that the variable so declared will be heavily used, so that its value can be stored in a CPU register. The hint can be ignored, and in most implementations it will be ignored if the address of the variable is taken. This use is deprecated.

(until C++17)

[edit] Static storage duration

A variable satisfying all following conditions has static storage duration :

  • It belongs to a namespace scope or are first declared with static or extern.
  • It does not have thread storage duration.
(since C++11)

The storage for these entities lasts for the duration of the program.

Thread storage duration

All variables declared with thread_local have thread storage duration.

The storage for these entities lasts for the duration of the thread in which they are created. There is a distinct object or reference per thread, and use of the declared name refers to the entity associated with the current thread.

(since C++11)

[edit] Automatic storage duration

The following variables have automatic storage duration :

  • Variables that belong to a block scope and are not explicitly declared static, thread_local,(since C++11) or extern. The storage for such variables lasts until the block in which they are created exits.
  • Variables that belong to a parameter scope (i.e. function parameters). The storage for a function parameter lasts until immediately after its destruction.

[edit] Dynamic storage duration

Objects created by the following methods during program execution have dynamic storage duration :

[edit] Linkage

A name can have external linkage , module linkage(since C++20), internal linkage, or no linkage:

  • An entity whose name has module linkage can be redeclared in another translation unit, as long as the redeclaration is attached to the same module.
(since C++20)
  • An entity whose name has internal linkage can be redeclared in another scope in the same translation unit.
  • An entity whose name has no linkage can only be redeclared in the same scope.

The following linkages are recognized:

[edit] No linkage

Any of the following names declared at block scope have no linkage:

  • variables that are not explicitly declared extern (regardless of the static modifier);
  • local classes and their member functions;
  • other names declared at block scope such as typedefs, enumerations, and enumerators.

Names not specified with external, module,(since C++20) or internal linkage also have no linkage, regardless of which scope they are declared in.

[edit] Internal linkage

Any of the following names declared at namespace scope have internal linkage:

  • variables, variable templates(since C++14), functions, or function templates declared static;
  • non-template (since C++14)variables of non-volatile const-qualified type, unless
  • they are inline,
(since C++17)
(since C++20)
  • they are explicitly declared extern, or
  • they were previously declared and the prior declaration did not have internal linkage;

In addition, all names declared in unnamed namespaces or a namespace within an unnamed namespace, even ones explicitly declared extern, have internal linkage.

(since C++11)

[edit] External linkage

Variables and functions with external linkage also have language linkage, which makes it possible to link translation units written in different programming languages.

Any of the following names declared at namespace scope have external linkage, unless they are declared in an unnamed namespace or their declarations are attached to a named module and are not exported(since C++20):

  • variables and functions not listed above (that is, functions not declared static, non-const variables not declared static, and any variables declared extern);
  • enumerations;
  • names of classes, their member functions, static data members (const or not), nested classes and enumerations, and functions first introduced with friend declarations inside class bodies;
  • names of all templates not listed above (that is, not function templates declared static).

Any of the following names first declared at block scope have external linkage:

  • names of variables declared extern;
  • names of functions.

Module linkage

Names declared at namespace scope have module linkage if their declarations are attached to a named module and are not exported, and do not have internal linkage.

(since C++20)

[edit] Static block variables

Block variables with static or thread(since C++11) storage duration are initialized the first time control passes through their declaration (unless their initialization is zero- or constant-initialization, which can be performed before the block is first entered). On all further calls, the declaration is skipped.

  • If the initialization throws an exception, the variable is not considered to be initialized, and initialization will be attempted again the next time control passes through the declaration.
  • If the initialization recursively enters the block in which the variable is being initialized, the behavior is undefined.
  • If multiple threads attempt to initialize the same static local variable concurrently, the initialization occurs exactly once (similar behavior can be obtained for arbitrary functions with std::call_once).
  • Usual implementations of this feature use variants of the double-checked locking pattern, which reduces runtime overhead for already-initialized local statics to a single non-atomic boolean comparison.
(since C++11)

The destructor for a block variable with static storage duration is called at program exit, but only if the initialization took place successfully.

Variables with static storage duration in all definitions of the same inline function (which may be implicitly inline) all refer to the same object defined in one translation unit, as long as the function has external linkage.

[edit] Translation-unit-local entities

The concept of translation-unit-local entities is standardized in C++20, see this page for more details.

An entity is translation-unit-local (or TU-local for short) if

  • it has a name with internal linkage, or
  • it does not have a name with linkage and is introduced within the definition of a TU-local entity, or
  • it is a template or template specialization whose template argument or template declaration uses a TU-local entity.

Bad things (usually violation of ODR) can happen if the type of a non-TU-local entity depends on a TU-local entity, or if a declaration of, or a deduction guide for,(since C++17) a non-TU-local entity names a TU-local entity outside its

  • function-body for a non-inline function or function template
  • initializer for a variable or variable template
  • friend declarations in a class definition
  • use of value of a variable, if the variable is usable in constant expressions

Such uses are disallowed in a module interface unit (outside its private-module-fragment, if any) or a module partition, and are deprecated in any other context.

A declaration that appears in one translation unit cannot name a TU-local entity declared in another translation unit that is not a header unit. A declaration instantiated for a template appears at the point of instantiation of the specialization.

(since C++20)

[edit] Notes

Names at the top-level namespace scope (file scope in C) that are const and not extern have external linkage in C, but internal linkage in C++.

Since C++11, auto is no longer a storage class specifier; it is used to indicate type deduction.

In C, the address of a register variable cannot be taken, but in C++, a variable declared register is semantically indistinguishable from a variable declared without any storage class specifiers.

(until C++17)

In C++, unlike C, variables cannot be declared register.

(since C++17)

Names of thread_local variables with internal or external linkage referred from different scopes may refer to the same or to different instances depending on whether the code is executing in the same or in different threads.

The extern keyword can also be used to specify language linkage and explicit template instantiation declarations, but it's not a storage class specifier in those cases (except when a declaration is directly contained in a language linkage specification, in which case the declaration is treated as if it contains the extern specifier).

Storage class specifiers, except for thread_local, are not allowed on explicit specializations and explicit instantiations:

template<class T>
struct S
{
    thread_local static int tlm;
};
 
template<>
thread_local int S<float>::tlm = 0; // "static" does not appear here

A const (may be implied by constexpr) variable template used to have internal linkage by default, which was inconsistent with other templated entities. Defect report CWG2387 corrected this.

(since C++14)
inline acts as a workaround for CWG2387 by giving external linkage by default. This is why the inline was added to many variable templates and then removed after having CWG2387 accepted. Standard library implementations also need to use inline as long as a supported compiler has not get CWG2387 implemented. See GCC Bugzilla #109126 and MSVC STL PR #4546. (since C++17)
Feature-test macro Value Std Feature
__cpp_threadsafe_static_init 200806L (C++11) Dynamic initialization and destruction with concurrency

[edit] Keywords

auto, register, static, extern, thread_local, mutable

[edit] Example

#include <iostream>
#include <mutex>
#include <string>
#include <thread>
 
thread_local unsigned int rage = 1;
std::mutex cout_mutex;
 
void increase_rage(const std::string& thread_name)
{
    ++rage; // modifying outside a lock is okay; this is a thread-local variable
    std::lock_guard<std::mutex> lock(cout_mutex);
    std::cout << "Rage counter for " << thread_name << ": " << rage << '\n';
}
 
int main()
{
    std::thread a(increase_rage, "a"), b(increase_rage, "b");
 
    {
        std::lock_guard<std::mutex> lock(cout_mutex);
        std::cout << "Rage counter for main: " << rage << '\n';
    }
 
    a.join();
    b.join();
}

Possible output:

Rage counter for a: 2
Rage counter for main: 1
Rage counter for b: 2

[edit] Defect reports

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

DR Applied to Behavior as published Correct behavior
CWG 216 C++98 unnamed class and enumeration in class scope have
different linkage from those in namespace scope
they all have external
linkage in these scopes
CWG 389 C++98 a name with no linkage should not be
used to declare an entity with linkage
a type without linkage shall not be used
as the type of a variable or function
with linkage, unless the variable
or function has C language linkage
CWG 426 C++98 an entity could be declared with both internal
and external linkage in the same translation unit
the program is ill-formed in this case
CWG 527 C++98 the type restriction introduced by the resolution of CWG
389 was also applied to variables and functions that
cannot be named outside their own translation units
the restriction is lifted for these
variables and functions (i.e. with no
linkage or internal linkage, or declared
within unnamed namespaces)
CWG 809 C++98 register served very little function deprecated
CWG 1648 C++11 static was implied even if
thread_local is combined with extern
implied only if no other storage
class specifier is present
CWG 1686 C++98
C++11
the name of a non-static variable declared in namespace
scope had internal linkage only if it is explicitly
declared const (C++98) or constexpr (C++11)
only required the type
to be const-qualified
CWG 2019 C++98 the storage duration of reference
members were unspecified
same as their complete object
CWG 2387 C++14 unclear whether const-qualified variable
template have internal linkage by default
const qualifier does not affect
the linkage of variable
templates or their instances
CWG 2533 C++98 the storage duration of implicitly-
created objects were unclear
made clear
CWG 2850 C++98 it was unclear when the storage for
function parameters are deallocated
made clear
CWG 2872 C++98 the meaning of “can be referred to” was unclear improved wording
P2788R0 C++20 declaring a const-qualified variable in a namespace
gave it internal linkage even in a module unit
internal linkage is not given

[edit] References

  • C++23 standard (ISO/IEC 14882:2024):
  • 6.7.5 Storage duration [basic.stc]
  • C++20 standard (ISO/IEC 14882:2020):
  • 6.7.5 Storage duration [basic.stc]
  • C++17 standard (ISO/IEC 14882:2017):
  • 6.7 Storage duration [basic.stc]
  • C++14 standard (ISO/IEC 14882:2014):
  • 3.7 Storage duration [basic.stc]
  • C++11 standard (ISO/IEC 14882:2011):
  • 3.7 Storage duration [basic.stc]
  • C++03 standard (ISO/IEC 14882:2003):
  • 3.7 Storage duration [basic.stc]
  • C++98 standard (ISO/IEC 14882:1998):
  • 3.7 Storage duration [basic.stc]

[edit] See also

C documentation for storage duration