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 | ||||||||||||||||
|
The storage class specifiers are a part of thedecl-specifier-seq of a name'sdeclaration syntax. Together with thescope of the name, they control two independent properties of the name: itsstorage duration and itslinkage.
Contents |
Thestorage duration is the property of anobject 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:
| (since C++11) |
Static, thread,(since C++11) and automatic storage durations are associated with objects introduced bydeclarations and withtemporary objects. The dynamic storage duration is associated with objects created by anew expression or withimplicitly created objects.
The storage duration categories apply to references as well.
The storage duration ofsubobjects and reference members is that of their complete object.
The following keywords arestorage class specifiers :
| (until C++11) |
| (until C++17) |
| (since C++11) |
In adecl-specifier-seq, there can be at most one storage class specifier, except thatthread_local may appear withstatic orextern(since C++11).
mutable has no effect on storage duration. For its usage, seeconst/volatile.
Other storage class specifiers can appear in thedecl-specifier-seq s of the following declarations:
Specifier | Can appear in thedecl-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 withstatic.
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) |
A variable satisfying all following conditions hasstatic storage duration :
| (since C++11) |
The storage for these entities lasts for the duration of the program.
Thread storage durationAll variables declared withthread_local havethread 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) |
The following variables haveautomatic storage duration :
Objects created by the following methods during program execution havedynamic storage duration :
A name can haveexternal linkage ,module linkage(since C++20),internal linkage, orno linkage:
| (since C++20) |
The following linkages are recognized:
Any of the following names declared at block scope have no linkage:
Names not specified with external, module,(since C++20) or internal linkage also have no linkage, regardless of which scope they are declared in.
Any of the following names declared at namespace scope have internal linkage:
| (since C++17) |
| (since C++20) |
In addition, all names declared inunnamed namespaces or a namespace within an unnamed namespace, even ones explicitly declaredextern, have internal linkage. | (since C++11) |
Variables and functions with external linkage also havelanguage 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):
Any of the following names first declared at block scope have external linkage:
Module linkageNames 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) |
This section is incomplete Reason: 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) |
Block variables with static or thread(since C++11) storage duration are initialized the first time control passes through their declaration (unless their initialization iszero- orconstant-initialization, which can be performed before the block is first entered). On all further calls, the declaration is skipped.
| (since C++11) |
The destructor for a block variable with static storage durationis called at program exit, but only if the initialization took place successfully.
Variables with static storage duration in all definitions of the sameinline 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.
The concept of translation-unit-local entities is standardized in C++20, seethis page for more details.
An entity istranslation-unit-local (orTU-local for short) if
Bad things (usually violation ofODR) can happen if the type of a non-TU-local entity depends on a TU-local entity, or if a declaration of, or adeduction guide for,(since C++17) a non-TU-local entity names a TU-local entity outside its
Such uses are disallowed in amodule 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 atemplate appears at the point of instantiation of the specialization. | (since C++20) |
Names at the top-level namespace scope (file scope in C) that areconst and notextern 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 aregister variable cannot be taken, but in C++, a variable declaredregister is semantically indistinguishable from a variable declared without any storage class specifiers. | (until C++17) |
In C++, unlike C, variables cannot be declaredregister. | (since C++17) |
Names ofthread_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.
Theextern keyword can also be used to specifylanguage linkage andexplicit 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 theextern specifier).
Storage class specifiers, except forthread_local, are not allowed onexplicit specializations andexplicit instantiations:
template<class T>struct S{ thread_localstaticint tlm;}; template<>thread_localint S<float>::tlm=0;// "static" does not appear here
Aconst (may be implied byconstexpr) variable template used to have internal linkage by default, which was inconsistent with other templated entities. Defect reportCWG2387 corrected this. | (since C++14) |
inline acts as a workaround forCWG2387 by giving external linkage by default. This is why theinline wasadded to many variable templates and thenremoved after having CWG2387 accepted. Standard library implementations also need to useinline as long as a supported compiler has not get CWG2387 implemented. SeeGCC Bugzilla #109126 andMSVC 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 |
auto,register,static,extern,thread_local,mutable
#include <iostream>#include <mutex>#include <string>#include <thread> thread_localunsignedint rage=1;std::mutex cout_mutex; void increase_rage(conststd::string& thread_name){++rage;// modifying outside a lock is okay; this is a thread-local variablestd::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: 2Rage counter for main: 1Rage counter for b: 2
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 withextern | 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 declaredconst (C++98) orconstexpr (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 |
C documentation forstorage duration |