- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
当前的标准草案(大概是 C++17)在 [basic.compound/4] 中说:
[ Note: An array object and its first element are not pointer-interconvertible, even though they have the same address. — end note ]
所以指向对象的指针不能是reinterpret_cast
'd 获取其封闭的数组指针。
现在,有 std::launder
, [ptr.launder/1] :
template<class T> [[nodiscard]] constexpr T* launder(T* p) noexcept
;Requires:
p
represents the address A of a byte in memory. An object X that is within its lifetime and whose type is similar to T is located at the address A. All bytes of storage that would be reachable through the result are reachable throughp
(see below).
可达的定义在[ptr.launder/3] :
Remarks: An invocation of this function may be used in a core constant expression whenever the value of its argument may be used in a core constant expression. A byte of storage is reachable through a pointer value that points to an object Y if it is within the storage occupied by Y, an object that is pointer-interconvertible with Y, or the immediately-enclosing array object if Y is an array element. The program is ill-formed if T is a function type or cv void.
现在,乍一看,似乎 std::launder
is 可用于进行上述转换,因为我已经强调了部分。
但是。如果 p
指向数组的一个对象,根据这个定义,数组的字节是可达的(即使 p
不是指针互转换为数组指针),就像洗牌的结果一样.所以,这个定义似乎没有说明这个问题。
所以,可以std::launder
用于将对象指针转换为其封闭的数组指针?
最佳答案
这取决于封闭的数组对象是否是一个完整的对象,如果不是,您是否可以通过指向该封闭数组对象的指针有效地访问更多字节(例如,因为它本身就是一个数组元素,或者指针与一个更大的对象,或指针可与作为数组元素的对象相互转换)。 “可达”要求意味着您不能使用 launder
来获取一个指针,该指针允许您访问比源指针值允许的更多字节,因为未定义行为的痛苦。这样可以保证某些未知代码调用launder
的可能性不会影响编译器的逃逸分析。
我想一些例子会有所帮助。 reinterpret_cast
s 下面的每个例子都是一个 int*
指向一个由 10 个 int
s 组成的数组的第一个元素到一个 int(*)[ 10]
。由于它们不是指针可互转换的,因此 reinterpret_cast
不会更改指针值,并且您会得到一个 int(*)[10]
,其值为“指向(无论数组是什么)的第一个元素”。然后,每个示例都尝试通过在强制转换指针上调用 std::launder
来获取指向整个数组的指针。
int x[10];
auto p = std::launder(reinterpret_cast<int(*)[10]>(&x[0]));
没关系;您可以通过源指针访问 x
的所有元素,而 launder
的结果不允许您访问其他任何内容。
int x2[2][10];
auto p2 = std::launder(reinterpret_cast<int(*)[10]>(&x2[0][0]));
这是未定义的。您只能通过源指针访问 x2[0]
的元素,但结果(将是指向 x2[0]
的指针)将允许您访问x2[1],你不能通过源码。
struct X { int a[10]; } x3, x4[2]; // assume no padding
auto p3 = std::launder(reinterpret_cast<int(*)[10]>(&x3.a[0])); // OK
没关系。同样,您无法通过指向 x3.a
的指针访问您已经无法访问的任何字节。
auto p4 = std::launder(reinterpret_cast<int(*)[10]>(&x4[0].a[0]));
这是(打算)未定义的。您将能够从结果中访问 x4[1]
,因为 x4[0].a
可以与 x4[0]
指针互转换>,所以指向前者的指针可以通过 reinterpret_cast
产生指向后者的指针,然后可以将其用于指针运算。见 https://wg21.link/LWG2859 .
struct Y { int a[10]; double y; } x5;
auto p3 = std::launder(reinterpret_cast<int(*)[10]>(&x5.a[0]));
这又是未定义的,因为您可以从结果指针到达 x5.y
(通过 reinterpret_cast
到 Y*
) 但是源指针不能用来访问它。
关于c++ - 可以使用 std::launder 将对象指针转换为其封闭数组指针吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51552713/
我正在开发一个小型图书馆,我需要做的一件事是让访问者访问一些数据并返回结果。 在一些较旧的 C++ 代码中,访问者需要声明一个 typedef return_type .例如,boost::stati
我正在尝试使用std:map类型的键和值制作std::any Visual Studio 2017 std::map m("lastname", "Ivanov"); std::cout (m["la
我已经在 C++ 的 map 中声明了一个集合为 std::map> .如何循环访问或打印设定值? 最佳答案 如果你知道如何迭代 std::map或 std::set单独地,您应该可以毫无问题地组合迭
如何循环? 我已经试过了: //----- code std::vector >::iterator it; for ( it = users.begin(); it != users.end();
我有两个用例。 A.我想同步访问两个线程的队列。 B.我想同步两个线程对队列的访问并使用条件变量,因为其中一个线程将等待另一个线程将内容存储到队列中。 对于用例 A,我看到了使用 std::lock_
我正在查看这两种类型特征的文档,但不确定有什么区别。我不是语言律师,但据我所知,它们都适用于“memcpy-able”类型。 它们可以互换使用吗? 最佳答案 不,这些术语不能互换使用。这两个术语都表示
我有以下测试代码,其中有一个参数 fS,它是 ofstream 的容器: #include #include #include #include int
这是这个问题的延续 c++ function ptr in unorderer_map, compile time error 我试图使用 std::function 而不是函数指针,并且只有当函数是
std::unordered_map str_bool_map = { {"a", true}, {"b", false}, {"c", true} }; 我们可以在此映射上使
我有以下对象 std::vector> vectorList; 然后我添加到这个使用 std::vector vec_tmp; vec_tmp.push_back(strDRG); vec_tmp.p
为什么 std::initializer_list不支持std::get<> , std::tuple_size和 std::tuple_element ?在constexpr中用得很多现在的表达式,
我有一个像这样定义的变量 auto drum = std::make_tuple ( std::make_tuple ( 0.3f , Ex
假设我有一个私有(private)std::map在我的类(class)里std::map 。我怎样才能将其转换为std::map返回给用户?我想要下面的原型(prototype) const std
假设我有一个私有(private)std::map在我的类(class)里std::map 。我怎样才能将其转换为std::map返回给用户?我想要下面的原型(prototype) const std
问题 我正在尝试将 lambda 闭包传递给 std::thread,它使用任意封闭参数调用任意封闭函数。 template std::thread timed_thread(Function&& f
我想创建一个模板类,可以容纳容器和容器的任意组合。例如,std::vector或 std::map ,例如。 我尝试了很多组合,但我必须承认模板的复杂性让我不知所措。我编译的关闭是这样的: templ
我有一个 std::vector>我将其分配给相同类型的第二个 vector 。 我收到这个编译器错误: /opt/gcc-8.2.0/include/c++/8.2.0/bits/stl_algob
有时候,我们有一个工厂可以生成一个 std::unique_ptr vector ,后来我们想在类/线程/你命名的之间共享这些指针。因此,最好改用 std::shared_ptr 。当然有一种方法可以
这个问题在这里已经有了答案: Sorting a vector of custom objects (14 个答案) 关闭 6 年前。 我创建了一个 vector vector ,我想根据我定义的参
我有三个类(class)成员: public: std::vector > getObjects(); std::vector > getObjects() const; privat
我是一名优秀的程序员,十分优秀!