Namespaces
Variants
Views
Actions

Difference between revisions of "cpp/language/reinterpret cast"

From cppreference.com
< cpp‎ | language
(fmt)
(+References)
 
(12 intermediate revisions by 8 users not shown)
Line 6: Line 6:
 
===Syntax===
 
===Syntax===
 
{{sdsc begin}}
 
{{sdsc begin}}
{{sdsc|{{ttb|reinterpret_cast<}} {{spar|new-type}} {{ttb|>(}} {{spar|expression}} {{ttb|)}}}}
+
{{sdsc|{{ttb|reinterpret_cast<}} {{spar|target-type}} {{ttb|>(}} {{spar|expression}} {{ttb|)}}}}
 
{{sdsc end}}
 
{{sdsc end}}
  
Returns a value of type {{spar|new-type}}.
+
Returns a value of type {{spar|target-type}}.
  
 
===Explanation===
 
===Explanation===
Unlike {{c/core|static_cast}}, but like {{c/core|const_cast}}, the {{c/core|reinterpret_cast}} expression does not compile to any CPU instructions (except when converting between integers and pointers or on obscure architectures where pointer representation depends on its type). It is purely a compile-time directive which instructs the compiler to treat {{spar|expression}} as if it had the type {{spar|new-type}}.
+
Unlike {{c/core|static_cast}}, but like {{c/core|const_cast}}, the {{c/core|reinterpret_cast}} expression does not compile to any CPU instructions (except when converting between integers and pointers, or between pointers on obscure architectures where pointer representation depends on its type). It is primarily a compile-time directive which instructs the compiler to treat {{spar|expression}} as if it had the type {{spar|target-type}}.
  
Only the following conversions can be done with {{c/core|reinterpret_cast}}, except when such conversions would cast away ''constness'' or ''volatility''.
+
Only the following conversions can be done with {{c/core|reinterpret_cast}}, except when such conversions would {{rlp|const_cast#Casting away constness|cast away constness}} (or volatility).
  
 
@1@ An expression of integral, enumeration, pointer, or pointer-to-member type can be converted to its own type. The resulting value is the same as the value of {{spar|expression}}.
 
@1@ An expression of integral, enumeration, pointer, or pointer-to-member type can be converted to its own type. The resulting value is the same as the value of {{spar|expression}}.
Line 23: Line 23:
 
@4@ Any value of type {{lc|std::nullptr_t}}, including {{c|nullptr}} can be converted to any integral type as if it were {{c|(void*)0}}, but no value, not even {{c|nullptr}} can be converted to {{lc|std::nullptr_t}}: {{c/core|static_cast}} should be used for that purpose.
 
@4@ Any value of type {{lc|std::nullptr_t}}, including {{c|nullptr}} can be converted to any integral type as if it were {{c|(void*)0}}, but no value, not even {{c|nullptr}} can be converted to {{lc|std::nullptr_t}}: {{c/core|static_cast}} should be used for that purpose.
 
}}
 
}}
@5@ Any object pointer type {{tt|T1*}} can be converted to another object pointer type {{tt|''cv'' T2*}}. This is exactly equivalent to {{box|{{c/core|static_cast<}}''cv''{{c/core| T2*>(static_cast<}}''cv''{{c/core| void*>(}}{{spar|expression}}{{c/core|))}}}} (which implies that if {{tt|T2}}'s alignment requirement is not stricter than {{tt|T1}}'s, the value of the pointer does not change and conversion of the resulting pointer back to its original type yields the original value). In any case, the resulting pointer may only be dereferenced safely if allowed by the ''type aliasing'' rules (see below).{{mark unreviewed dr|cwg}}<!-- cwg1412 -->
+
@5@ Any object pointer type {{tt|T1*}} can be converted to another object pointer type {{tt|''cv'' T2*}}. This is exactly equivalent to {{box|{{c/core|static_cast<}}''cv''{{c/core| T2*>(static_cast<}}''cv''{{c/core| void*>(}}{{spar|expression}}{{c/core|))}}}} (which implies that if {{tt|T2}}'s alignment requirement is not stricter than {{tt|T1}}'s, the value of the pointer does not change and conversion of the resulting pointer back to its original type yields the original value). In any case, the resulting pointer may only be dereferenced safely if the dereferenced value is [[#Type accessibility|type-accessible]].{{mark unreviewed dr|cwg}}<!--cwg1412-->
@6@ An {{rev inl|until=c++11|{{rlp|value category#lvalue|lvalue}}}}{{rev inl|since=c++11|{{rlp|value category#glvalue|glvalue}}}} expression of type {{tt|T1}} can be converted to reference to another type {{tt|T2}}. The result is that of {{c|*reinterpret_cast<T2*>(p)}}, where {{c|p}} is a pointer of type “pointer to {{tt|T1}}” to the object designated by {{spar|expression}}. No temporary is created, no copy is made, no constructors or conversion functions are called. The resulting reference can only be accessed safely if allowed by the ''type aliasing'' rules (see below).
+
@6@ An {{rev inl|until=c++11|{{rlp|value category#lvalue|lvalue}}}}{{rev inl|since=c++11|{{rlp|value category#glvalue|glvalue}}}} expression of type {{tt|T1}} can be converted to reference to another type {{tt|T2}}. The result is that of {{c|*reinterpret_cast<T2*>(p)}}, where {{c|p}} is a pointer of type “pointer to {{tt|T1}}” to the object or function designated by {{spar|expression}}. No temporary is created, no copy is made, no constructors or conversion functions are called. The resulting reference can only be accessed safely if it is [[#Type accessibility|type-accessible]].
@7@ Any pointer to function can be converted to a pointer to a different function type. Calling the function through a pointer to a different function type is undefined, but converting such pointer back to pointer to the original function type yields the pointer to the original function.
+
@7@ Any pointer to function can be converted to a pointer to a different function type. The result is unspecified, but converting such pointer back to pointer to the original function type yields the pointer to the original function. The resulting pointer can only be called safely if it function type is [[#Call compatibility|call-compatible]] with the original function type.
@8@ On some implementations (in particular, on any POSIX compatible system as required by [http://pubs.opengroup.org/onlinepubs/9699919799/functions/dlsym.html {{tt|dlsym}}]), a function pointer can be converted to {{c/core|void*}} or any other object pointer, or vice versa. If the implementation supports conversion in both directions, conversion to the original type yields the original value, otherwise the resulting pointer cannot be dereferenced or called safely.
+
@8@ On some implementations (in particular, on any POSIX compatible system as required by [https://pubs.opengroup.org/onlinepubs/9699919799/functions/dlsym.html {{tt|dlsym}}]), a function pointer can be converted to {{c/core|void*}} or any other object pointer, or vice versa. If the implementation supports conversion in both directions, conversion to the original type yields the original value, otherwise the resulting pointer cannot be dereferenced or called safely.
 
@9@ The null pointer value of any pointer type can be converted to any other pointer type, resulting in the null pointer value of that type. Note that the null pointer constant {{c|nullptr}} or any other value of type {{lc|std::nullptr_t}} cannot be converted to a pointer with {{c/core|reinterpret_cast}}: implicit conversion or {{c/core|static_cast}} should be used for this purpose.
 
@9@ The null pointer value of any pointer type can be converted to any other pointer type, resulting in the null pointer value of that type. Note that the null pointer constant {{c|nullptr}} or any other value of type {{lc|std::nullptr_t}} cannot be converted to a pointer with {{c/core|reinterpret_cast}}: implicit conversion or {{c/core|static_cast}} should be used for this purpose.
 
@10@ A pointer to member function can be converted to pointer to a different member function of a different type. Conversion back to the original type yields the original value, otherwise the resulting pointer cannot be used safely.
 
@10@ A pointer to member function can be converted to pointer to a different member function of a different type. Conversion back to the original type yields the original value, otherwise the resulting pointer cannot be used safely.
Line 33: Line 33:
 
{{cpp/language/cast return}}
 
{{cpp/language/cast return}}
  
===Keywords===
+
===Type aliasing===
{{ltt|cpp/keyword/reinterpret_cast}}
+
====Type accessibility====
 
+
If a type {{tt|T_ref}} is {{rlp|implicit conversion#Similar types|similar}} to any of the following types, an object of {{rlpsd|type#Dynamic type}} {{tt|T_obj}} is ''type-accessible'' through a {{rev inl|until=c++11|lvalue}}{{rev inl|since=c++11|glvalue}} of type {{tt|T_ref}}:
=== Type aliasing ===
+
* {{c/core|char}}
Whenever an attempt is made to read or modify the stored value of an object of type {{tt|DynamicType}} through a glvalue of type {{tt|AliasedType}}, the behavior is undefined unless one of the following is true:
+
* {{c/core|unsigned char}}
* {{tt|AliasedType}} and {{tt|DynamicType}} are ''similar''.
+
{{rrev|since=c++17|
* {{tt|AliasedType}} is the (possibly {{rlp|cv}}-qualified) signed or unsigned variant of {{tt|DynamicType}}.
+
* {{lc|std::byte}}
* {{tt|AliasedType}} is {{rev inl|since=c++17|{{ltt|cpp/types/byte|std::byte}},}} {{c/core|char}}, or {{c/core|unsigned char}}: this is intended to permit examination of the  {{rlp|object#Object representation and value representation|object representation}} of any object as an array of bytes.
+
 
+
Informally, two types are ''similar'' if, ignoring top-level cv-qualification:
+
* they are the same type; or
+
* they are both pointers, and the pointed-to types are similar; or
+
* they are both pointers to member of the same class, and the types of the pointed-to members are similar; or
+
{{rrev multi|until1=c++20|rev1=
+
* they are both arrays of the same size or both arrays of unknown bound, and the array element types are similar.
+
|rev2=
+
* they are both arrays of the same size or at least one of them is array of unknown bound, and the array element types are similar.
+
 
}}
 
}}
 +
* {{tt|T_obj}}
 +
* the signed or unsigned type corresponding to {{tt|T_obj}}
  
For example:
+
If a program attempts to read or modify the stored value of an object through a {{rev inl|until=c++11|lvalue}}{{rev inl|since=c++11|glvalue}} through which it is not type-accessible, the behavior is undefined.
*{{c/core|const int * volatile *}} and {{c/core|int * * const}} are similar;
+
*{{c/core|const int (* volatile S::* const)[20]}} and {{c/core|int (* const S::* volatile)[20]}} are similar;
+
*{{c/core|int (* const *)(int *)}} and {{c/core|int (* volatile *)(int *)}} are similar;
+
*{{c/core|int (S::*)() const}} and {{c/core|int (S::*)()}} are ''not'' similar;
+
*{{c/core|int (*)(int *)}} and {{c/core|int (*)(const int *)}} are ''not'' similar;
+
*{{c/core|const int (*)(int *)}} and {{c/core|int (*)(int *)}} are ''not'' similar;
+
*{{c/core|int (*)(int * const)}} and {{c/core|int (*)(int *)}} are similar (they are the same type);
+
*{{c/core|std::pair<int, int>}} and {{c/core|std::pair<const int, int>}} are ''not'' similar.
+
  
 
This rule enables type-based alias analysis, in which a compiler assumes that the value read through a glvalue of one type is not modified by a write to a glvalue of a different type (subject to the exceptions noted above).
 
This rule enables type-based alias analysis, in which a compiler assumes that the value read through a glvalue of one type is not modified by a write to a glvalue of a different type (subject to the exceptions noted above).
  
 
Note that many C++ compilers relax this rule, as a non-standard language extension, to allow wrong-type access through the inactive member of a {{rlp|union}} (such access is not undefined in C).
 
Note that many C++ compilers relax this rule, as a non-standard language extension, to allow wrong-type access through the inactive member of a {{rlp|union}} (such access is not undefined in C).
 +
 +
====Call compatibility====
 +
If any of the following conditions is satisfied, a type {{tt|T_call}} is ''call-compatible'' with a function type {{tt|T_func}}:
 +
* {{tt|T_call}} is the same type as {{tt|T_func}}.
 +
{{rrev|since=c++17|
 +
* {{tt|T_func*}} can be converted to {{tt|T_call*}} via a {{rlp|implicit conversion#Function pointer conversions|function pointer conversion}}.
 +
}}
 +
 +
If a function is called through an expression whose {{rlpsd|function#Function type}} is not call-compatible with the type of the called function’s definition, the behavior is undefined.
  
 
===Notes===
 
===Notes===
Line 130: Line 123:
 
std::memcpy(&n, &d, sizeof d);              // OK
 
std::memcpy(&n, &d, sizeof d);              // OK
 
n = std::bit_cast<std::int64_t>(d);          // also OK
 
n = std::bit_cast<std::int64_t>(d);          // also OK
}}
 
 
Accessing the object representation of an object through an aliased pointer to {{rev inl|since=c++17|{{ltt|cpp/types/byte|std::byte}},}} {{c/core|char}}, or {{c/core|unsigned char}} can be UB. In the following example, the {{c|reinterpret_cast}} is exactly equivalent to {{c|static_cast<unsigned char*>(static_cast<void*>(n))}} and as such, the value of the pointer is unchanged and therefore it
 
points to the original object (the {{c|int}}) (see [[cpp/language/static_cast|static_cast]]). When a is dereferenced, the behaviour is undefined because the value of the resulting expression would not be the value of the first byte, but the value of the whole {{c|int}} object (123456), which is not a value representable by {{c|unsigned char}}.
 
 
Further, even if we ignore this issue, a does not point to an array of {{c|unsigned char}}, because such an array has never been created, and therefore pointer arithmetics on a is undefined behaviour. An object representation as defined by is merely a sequence of {{c|unsigned char}} objects, not an array, and is therefore unsuitable for pointer arithmetics. No array is ever created explicitly, and no operation is being called in the above code that would implicitly create an array, since casts are not operations that implicitly create objects.
 
 
{{source|1=
 
void print_hex(int n) {
 
// unsigned char* a = reinterpret_cast<unsigned char*>(&n); // Defined per the strict aliasing rule but causes UB when the pointer is accessed
 
unsigned char* a = std::start_lifetime_as<unsigned char[sizeof(int)]>(&n); // OK
 
for (int i = 0; i < sizeof(int); ++i)
 
printf("%02x ", a[i]);
 
}
 
 
int main() {
 
print_hex(123456);
 
}
 
 
}}
 
}}
  
Line 154: Line 129:
 
}}
 
}}
  
The paragraph defining the strict aliasing rule in the standard used to contain two additional bullets partially inherited from C:
+
In C, aggregate copy and assignment access the aggregate object as a whole. But in C++ such actions are always performed through a member function call, which accesses the individual subobjects rather than the entire object (or, in the case of unions, copies the object representation, i.e., via {{c/core|unsigned char}}).
* {{tt|AliasedType}} is an {{rlp|aggregate initialization|aggregate type}} or a {{rlp|union}} type which holds one of the aforementioned types as an element or non-static member (including, recursively, elements of subaggregates and non-static data members of the contained unions).
+
* {{tt|AliasedType}} is a (possibly {{rlp|cv}}-qualified) {{rlp|derived class|base class}} of {{tt|DynamicType}}.
+
  
These bullets describe situations that cannot arise in C++ and therefore are omitted from the discussion above. In C, aggregate copy and assignment access the aggregate object as a whole. But in C++ such actions are always performed through a member function call, which accesses the individual subobjects rather than the entire object (or, in the case of unions, copies the object representation, i.e., via {{c/core|unsigned char}}). These bullets were eventually removed via {{cwg|2051}}.
+
===Keywords===
 +
{{ltt|cpp/keyword/reinterpret_cast}}
  
 
===Example===
 
===Example===
Line 208: Line 182:
 
}}
 
}}
  
=== Defect reports ===
+
===Defect reports===
 
{{dr list begin}}
 
{{dr list begin}}
 
{{dr list item|wg=cwg|dr=195|std=C++98|before=conversion between function pointers<br>and object pointers not allowed|after=made conditionally-supported}}
 
{{dr list item|wg=cwg|dr=195|std=C++98|before=conversion between function pointers<br>and object pointers not allowed|after=made conditionally-supported}}
Line 214: Line 188:
 
{{dr list item|wg=cwg|dr=799|std=C++98|before=it was unclear which identity conversion<br>can be done by {{c/core|reinterpret_cast}}|after=made clear}}
 
{{dr list item|wg=cwg|dr=799|std=C++98|before=it was unclear which identity conversion<br>can be done by {{c/core|reinterpret_cast}}|after=made clear}}
 
{{dr list item|wg=cwg|dr=1268|std=C++11|before={{c/core|reinterpret_cast}} could only cast lvalues to reference types|after=xvalues also allowed}}
 
{{dr list item|wg=cwg|dr=1268|std=C++11|before={{c/core|reinterpret_cast}} could only cast lvalues to reference types|after=xvalues also allowed}}
 +
{{dr list item|wg=cwg|dr=2780|std=C++98|before={{c/core|reinterpret_cast}} could not cast<br>function lvalues to other reference types|after=allowed}}
 
{{dr list end}}
 
{{dr list end}}
 +
 +
===References===
 +
{{ref std c++23}}
 +
{{ref std|section=7.6.1.10|title=Reinterpret cast|id=expr.reinterpret.cast}}
 +
{{ref std end}}
 +
{{ref std c++20}}
 +
{{ref std|section=7.6.1.9|title=Reinterpret cast|id=expr.reinterpret.cast}}
 +
{{ref std end}}
 +
{{ref std c++17}}
 +
{{ref std|section=8.2.10|title=Reinterpret cast|id=expr.reinterpret.cast}}
 +
{{ref std end}}
 +
{{ref std c++14}}
 +
{{ref std|section=5.2.10|title=Reinterpret cast|id=expr.reinterpret.cast}}
 +
{{ref std end}}
 +
{{ref std c++11}}
 +
{{ref std|section=5.2.10|title=Reinterpret cast|id=expr.reinterpret.cast}}
 +
{{ref std end}}
 +
{{ref std c++98}}
 +
{{ref std|section=5.2.10|title=Reinterpret cast|id=expr.reinterpret.cast}}
 +
{{ref std end}}
 +
{{ref std c++03}}
 +
{{ref std|section=5.2.10|title=Reinterpret cast|id=expr.reinterpret.cast}}
 +
{{ref std end}}
  
 
===See also===
 
===See also===

Latest revision as of 23:38, 13 August 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
 
 

Converts between types by reinterpreting the underlying bit pattern.

Contents

[edit] Syntax

reinterpret_cast< target-type >( expression )

Returns a value of type target-type.

[edit] Explanation

Unlike static_cast, but like const_cast, the reinterpret_cast expression does not compile to any CPU instructions (except when converting between integers and pointers, or between pointers on obscure architectures where pointer representation depends on its type). It is primarily a compile-time directive which instructs the compiler to treat expression as if it had the type target-type.

Only the following conversions can be done with reinterpret_cast, except when such conversions would cast away constness (or volatility).

1) An expression of integral, enumeration, pointer, or pointer-to-member type can be converted to its own type. The resulting value is the same as the value of expression.
2) A pointer can be converted to any integral type large enough to hold all values of its type (e.g. to std::uintptr_t).
3) A value of any integral or enumeration type can be converted to a pointer type. A pointer converted to an integer of sufficient size and back to the same pointer type is guaranteed to have its original value, otherwise the resulting pointer cannot be dereferenced safely (the round-trip conversion in the opposite direction is not guaranteed; the same pointer may have multiple integer representations) The null pointer constant NULL or integer zero is not guaranteed to yield the null pointer value of the target type; static_cast or implicit conversion should be used for this purpose.
4) Any value of type std::nullptr_t, including nullptr can be converted to any integral type as if it were (void*)0, but no value, not even nullptr can be converted to std::nullptr_t: static_cast should be used for that purpose.
(since C++11)
5) Any object pointer type T1* can be converted to another object pointer type cv T2*. This is exactly equivalent to static_cast<cv T2*>(static_cast<cv void*>(expression)) (which implies that if T2's alignment requirement is not stricter than T1's, the value of the pointer does not change and conversion of the resulting pointer back to its original type yields the original value). In any case, the resulting pointer may only be dereferenced safely if the dereferenced value is type-accessible.
6) An lvalue(until C++11)glvalue(since C++11) expression of type T1 can be converted to reference to another type T2. The result is that of *reinterpret_cast<T2*>(p), where p is a pointer of type “pointer to T1” to the object or function designated by expression. No temporary is created, no copy is made, no constructors or conversion functions are called. The resulting reference can only be accessed safely if it is type-accessible.
7) Any pointer to function can be converted to a pointer to a different function type. The result is unspecified, but converting such pointer back to pointer to the original function type yields the pointer to the original function. The resulting pointer can only be called safely if it function type is call-compatible with the original function type.
8) On some implementations (in particular, on any POSIX compatible system as required by dlsym), a function pointer can be converted to void* or any other object pointer, or vice versa. If the implementation supports conversion in both directions, conversion to the original type yields the original value, otherwise the resulting pointer cannot be dereferenced or called safely.
9) The null pointer value of any pointer type can be converted to any other pointer type, resulting in the null pointer value of that type. Note that the null pointer constant nullptr or any other value of type std::nullptr_t cannot be converted to a pointer with reinterpret_cast: implicit conversion or static_cast should be used for this purpose.
10) A pointer to member function can be converted to pointer to a different member function of a different type. Conversion back to the original type yields the original value, otherwise the resulting pointer cannot be used safely.
11) A pointer to member object of some class T1 can be converted to a pointer to another member object of another class T2. If T2's alignment is not stricter than T1's, conversion back to the original type T1 yields the original value, otherwise the resulting pointer cannot be used safely.

As with all cast expressions, the result is:

  • an lvalue if target-type is an lvalue reference type or an rvalue reference to function type(since C++11);
  • an xvalue if target-type is an rvalue reference to object type;
(since C++11)
  • a prvalue otherwise.

[edit] Type aliasing

[edit] Type accessibility

If a type T_ref is similar to any of the following types, an object of dynamic type T_obj is type-accessible through a lvalue(until C++11)glvalue(since C++11) of type T_ref:

  • char
  • unsigned char
  • std::byte
(since C++17)
  • T_obj
  • the signed or unsigned type corresponding to T_obj

If a program attempts to read or modify the stored value of an object through a lvalue(until C++11)glvalue(since C++11) through which it is not type-accessible, the behavior is undefined.

This rule enables type-based alias analysis, in which a compiler assumes that the value read through a glvalue of one type is not modified by a write to a glvalue of a different type (subject to the exceptions noted above).

Note that many C++ compilers relax this rule, as a non-standard language extension, to allow wrong-type access through the inactive member of a union (such access is not undefined in C).

[edit] Call compatibility

If any of the following conditions is satisfied, a type T_call is call-compatible with a function type T_func:

  • T_call is the same type as T_func.
(since C++17)

If a function is called through an expression whose function type is not call-compatible with the type of the called function’s definition, the behavior is undefined.

[edit] Notes

Assuming that alignment requirements are met, a reinterpret_cast does not change the value of a pointer outside of a few limited cases dealing with pointer-interconvertible objects:

struct S1 { int a; } s1;
struct S2 { int a; private: int b; } s2; // not standard-layout
union U { int a; double b; } u = {0};
int arr[2];
 
int* p1 = reinterpret_cast<int*>(&s1); // value of p1 is "pointer to s1.a" because
                                       // s1.a and s1 are pointer-interconvertible
 
int* p2 = reinterpret_cast<int*>(&s2); // value of p2 is unchanged by reinterpret_cast
                                       // and is "pointer to s2". 
 
int* p3 = reinterpret_cast<int*>(&u);  // value of p3 is "pointer to u.a":
                                       // u.a and u are pointer-interconvertible
 
double* p4 = reinterpret_cast<double*>(p3); // value of p4 is "pointer to u.b": u.a and
                                            // u.b are pointer-interconvertible because
                                            // both are pointer-interconvertible with u
 
int* p5 = reinterpret_cast<int*>(&arr); // value of p5 is unchanged by reinterpret_cast
                                        // and is "pointer to arr"

Performing a class member access that designates a non-static data member or a non-static member function on a glvalue that does not actually designate an object of the appropriate type - such as one obtained through a reinterpret_cast - results in undefined behavior:

struct S { int x; };
struct T { int x; int f(); };
struct S1 : S {};    // standard-layout
struct ST : S, T {}; // not standard-layout
 
S s = {};
auto p = reinterpret_cast<T*>(&s); // value of p is "pointer to s"
auto i = p->x; // class member access expression is undefined behavior;
               // s is not a T object
p->x = 1; // undefined behavior
p->f();   // undefined behavior
 
S1 s1 = {};
auto p1 = reinterpret_cast<S*>(&s1); // value of p1 is "pointer to the S subobject of s1"
auto i = p1->x; // OK
p1->x = 1;      // OK
 
ST st = {};
auto p2 = reinterpret_cast<S*>(&st); // value of p2 is "pointer to st"
auto i = p2->x; // undefined behavior
p2->x = 1;      // undefined behavior

Many compilers issue "strict aliasing" warnings in such cases, even though technically such constructs run afoul of something other than the paragraph commonly known as the "strict aliasing rule".

The purpose of strict aliasing and related rules is to enable type-based alias analysis, which would be decimated if a program can validly create a situation where two pointers to unrelated types (e.g., an int* and a float*) could simultaneously exist and both can be used to load or store the same memory (see this email on SG12 reflector). Thus, any technique that is seemingly capable of creating such a situation necessarily invokes undefined behavior.

When it is needed to interpret the bytes of an object as a value of a different type, std::memcpy or std::bit_cast(since C++20) can be used:

double d = 0.1;
std::int64_t n;
static_assert(sizeof n == sizeof d);
// n = *reinterpret_cast<std::int64_t*>(&d); // Undefined behavior
std::memcpy(&n, &d, sizeof d);               // OK
n = std::bit_cast<std::int64_t>(d);          // also OK

If the implementation provides std::intptr_t and/or std::uintptr_t, then a cast from a pointer to an object type or cv void to these types is always well-defined. However, this is not guaranteed for a function pointer.

(since C++11)

In C, aggregate copy and assignment access the aggregate object as a whole. But in C++ such actions are always performed through a member function call, which accesses the individual subobjects rather than the entire object (or, in the case of unions, copies the object representation, i.e., via unsigned char).

[edit] Keywords

reinterpret_cast

[edit] Example

Demonstrates some uses of reinterpret_cast:

#include <cassert>
#include <cstdint>
#include <iostream>
 
int f() { return 42; }
 
int main()
{
    int i = 7;
 
    // pointer to integer and back
    std::uintptr_t v1 = reinterpret_cast<std::uintptr_t>(&i); // static_cast is an error
    std::cout << "The value of &i is " << std::showbase << std::hex << v1 << '\n';
    int* p1 = reinterpret_cast<int*>(v1);
    assert(p1 == &i);
 
    // pointer to function to another and back
    void(*fp1)() = reinterpret_cast<void(*)()>(f);
    // fp1(); undefined behavior
    int(*fp2)() = reinterpret_cast<int(*)()>(fp1);
    std::cout << std::dec << fp2() << '\n'; // safe
 
    // type aliasing through pointer
    char* p2 = reinterpret_cast<char*>(&i);
    std::cout << (p2[0] == '\x7' ? "This system is little-endian\n"
                                 : "This system is big-endian\n");
 
    // type aliasing through reference
    reinterpret_cast<unsigned int&>(i) = 42;
    std::cout << i << '\n';
 
    [[maybe_unused]] const int &const_iref = i;
    // int &iref = reinterpret_cast<int&>(
    //     const_iref); // compiler error - can't get rid of const
    // Must use const_cast instead: int &iref = const_cast<int&>(const_iref);
}

Possible output:

The value of &i is 0x7fff352c3580
42
This system is little-endian
42

[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 195 C++98 conversion between function pointers
and object pointers not allowed
made conditionally-supported
CWG 658 C++98 the result of pointer conversions was unspecified
(except for conversions back to the original type)
specification provided for pointers
whose pointed-to types satisfy
the alignment requirements
CWG 799 C++98 it was unclear which identity conversion
can be done by reinterpret_cast
made clear
CWG 1268 C++11 reinterpret_cast could only cast lvalues to reference types xvalues also allowed
CWG 2780 C++98 reinterpret_cast could not cast
function lvalues to other reference types
allowed

[edit] References

  • C++23 standard (ISO/IEC 14882:2024):
  • 7.6.1.10 Reinterpret cast [expr.reinterpret.cast]
  • C++20 standard (ISO/IEC 14882:2020):
  • 7.6.1.9 Reinterpret cast [expr.reinterpret.cast]
  • C++17 standard (ISO/IEC 14882:2017):
  • 8.2.10 Reinterpret cast [expr.reinterpret.cast]
  • C++14 standard (ISO/IEC 14882:2014):
  • 5.2.10 Reinterpret cast [expr.reinterpret.cast]
  • C++11 standard (ISO/IEC 14882:2011):
  • 5.2.10 Reinterpret cast [expr.reinterpret.cast]
  • C++98 standard (ISO/IEC 14882:1998):
  • 5.2.10 Reinterpret cast [expr.reinterpret.cast]
  • C++03 standard (ISO/IEC 14882:2003):
  • 5.2.10 Reinterpret cast [expr.reinterpret.cast]

[edit] See also

const_cast conversion adds or removes const[edit]
static_cast conversion performs basic conversions[edit]
dynamic_cast conversion performs checked polymorphic conversions[edit]
explicit casts permissive conversions between types [edit]
standard conversions implicit conversions from one type to another[edit]
(C++20)
reinterpret the object representation of one type as that of another
(function template) [edit]