std::is_pointer_interconvertible_with_class

From cppreference.com
< cpp‎ | types
 
 
Metaprogramming library
Type traits
Type categories
(C++11)
(C++11)
(C++11)
(C++11)
(C++11)
(C++11)
(C++11)
(C++11)
Type properties
(C++11)
(C++11)
(C++14)
(C++11)
(C++11)(until C++20)
(C++11)(deprecated in C++20)
(C++11)
Type trait constants
Metafunctions
(C++17)
Supported operations
Relationships and property queries
Type modifications
(C++11)(C++11)(C++11)
Type transformations
(C++11)(deprecated in C++23)
(C++11)(deprecated in C++23)
(C++11)
(C++11)
(C++17)

(C++11)(until C++20)(C++17)
Compile-time rational arithmetic
Compile-time integer sequences
 
Defined in header <type_traits>
template<class S, class M>
constexpr bool is_pointer_interconvertible_with_class( M S::* mp ) noexcept;
(since C++20)

Given an object s of type S, determines whether s.*mp refers to a subobject of s and s is pointer-interconvertible with its subobject s.*mp. The program is ill-formed if S is not a complete type.

If S is not a StandardLayoutType, or M is not an object type, or mp is equal to nullptr, the result is always false.

Parameters

mp - a pointer-to-member to detect

Return value

true if s.*mp refers a subobject of s and s is pointer-interconvertible with its subobject s.*mp, otherwise false, where s is an object of type S.

Notes

The type of a pointer-to-member expression &S::m is not always M S::*, where m is of type M, because m may be a member inherited from a base class of S. The template arguments can be specified in order to avoid potentially surprising results.

If there is a value mp of type M S::* such that std::is_pointer_interconvertible_with_class(mp) == true, then reinterpret_cast<M&>(s) has well-defined result and it refers the same subobject as s.*mp, where s is a valid lvalue of type S.

On common platforms, the bit pattern of mp is all zero if std::is_pointer_interconvertible_with_class(mp) == true.

Feature-test macro: __cpp_lib_is_pointer_interconvertible

Example

#include <type_traits>
#include <iostream>
 
struct Foo { int x; };
struct Bar { int y; };
 
struct Baz : Foo, Bar {}; // not standard-layout
 
int main()
{
    std::cout << std::boolalpha
        << std::is_same_v<decltype(&Baz::x), int Baz::*> << '\n'
        << std::is_pointer_interconvertible_with_class(&Baz::x) << '\n'
        << std::is_pointer_interconvertible_with_class<Baz, int>(&Baz::x) << '\n';
}

Output:

false
true
false

See also

checks if a type is a standard-layout type
(class template)
checks if a type is a pointer to a non-static member object
(class template)