- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
placement new 的返回值与其操作数的强制转换值之间是否存在(语义)差异?
struct Foo { ... };
char buffer[...];
Foo *a = new(buffer) Foo;
Foo *b = reinterpret_cast<Foo *>(buffer);
a
和 b
有什么不同吗?
编辑:根据 DaBler 的评论,如果使用 const/reference 成员,这个问题表明存在差异:Placement new and assignment of class with const member
所以,我的一点更新问题:a
和 b
是否有任何不同,如果 Foo
没有 const 或引用成员?
最佳答案
仅限 a
可以安全地用于直接访问 Foo
由放置 new-expression 创建的对象(为了便于引用,我们将其称为 x
)。使用 b
需要 std::launder
.
a
的值在 [expr.new]/1 中指定:
If the entity is a non-array object, the result of the new-expression is a pointer to the object created.
a
的值因此是“指向 x
的指针”。当然,这个指针可以安全地用于访问 x
.
reinterpret_cast<Foo*>(buffer)
将数组到指针的转换应用于 buffer
(见 [expr.reinterpret.cast]/1)。应用转换后的结果值为“指向buffer
的第一个元素的指针”。这是 reinterpret_cast
一个对象指针指向不同类型的对象指针,定义为等价于 static_cast<Foo*>(static_cast<void*>(buffer))
由 [expr.reinterpret.cast]/7 .
内投到void*
实际上是一种隐式转换。根据 [conv.ptr]/2 ,
The pointer value is unchanged by this conversion.
因此,内部转换产生 void*
值为“指向buffer
的第一个元素的指针”。
外部 Actor 由 [expr.static.cast]/13 管理,我已将其重新格式化为要点:
A prvalue of type “pointer to cv1
void
” can be converted to a prvalue of type “pointer to cv2T
”, whereT
is an object type and cv2 is the same cv-qualification as, or greater cv-qualification than, cv1.
If the original pointer value represents the address
A
of a byte in memory andA
does not satisfy the alignment requirement ofT
, then the resulting pointer value is unspecified.Otherwise, if the original pointer value points to an object
a
, and there is an objectb
of typeT
(ignoring cv-qualification) that is pointer-interconvertible witha
, the result is a pointer tob
.Otherwise, the pointer value is unchanged by the conversion.
假设 buffer
适当对齐(如果不是,那么在此之前你会遇到麻烦),第一个项目符号不适用。第二个项目符号同样不适用,因为没有 pointer-interconvertiblity这里。因此,我们击中了第三个项目符号 - “指针值不会因转换而改变”,并且仍然是“指向 buffer
的第一个元素的指针”。
因此,b
不指向 Foo
对象 x
;它指向第一个 char
buffer
的元素, 即使它的类型是 Foo*
.因此它不能用于访问 x
;尝试这样做会产生未定义的行为(对于非静态数据成员的情况,[expr.ref] 的省略;对于非静态成员函数的情况,[class.mfct.non-static]/2 的省略)。
恢复指向 x
的指针来自 b
, std::launder
可以使用:
b = std::launder(b); // value of b is now "pointer to x"
// and can be used to access x
关于c++ - Placement new 的返回值与其操作数的强制转换值之间是否存在(语义)差异?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47653305/
int enter_path(char** path) { char* standard = "./questions.txt"; printf("\n\t%s\n\t%s",
我有以下几行代码: #define PORT 9987 和 char *ptr = (char *)&PORT; 这似乎适用于我的服务器代码。但是当我在我的客户端代码中写它时,它给出了这个错误信息:
大家好,我在成员函数中有以下内容 int tt = 6; vector>& temp = m_egressCandidatesByDestAndOtMode[tt]; set& egressCandi
我知道您可以通过以下方式在正则表达式中使用 NOT 操作数: [^AB] :匹配除 "A" 之外的任何内容或"B" A(?!B) :匹配"A" ,后面不跟 "B" (?
我的代码如下,下面还解释了为什么会发生左值; typedef struct test_item { char id[MENU_NAME_LEN + NULL_SPACE]; MenuF
我正在审查一些 javascript 代码,程序员在几个地方使用了 >>。我试图在谷歌上搜索但找不到这个操作数/运算符的作用。所以我来了。下面的代码示例: var triplet=(((binarra
我使用以下行(希望这是最佳实践,如果不正确请纠正我)来处理命令行选项: #!/usr/bin/bash read -r -d '' HELP &2 for i in "${invalid_opti
我正在尝试编辑一个计时器应用程序,出现了这行代码。我该如何解决? let styleMask: Int = NSClosableWindowMask | NSTitledWindowMask 错误是:
我可以得到两个特定日期之间的差异,这将等于日期总数。现在我想将工作日除以总天数并得到整数输出。 @IBAction func go(_ sender: UIButton) { let con
我的项目有一个问题,它应该使用一个线程将每一行相加,然后将它们全部相加,但是我收到一个错误,指出左值需要作为一元 '&"操作数 pthread_create(&tid, NULL, &sum_line
我的代码有问题。有以下功能: static Poly PolyFromCoeff(int coeff); static Mono MonoFromPoly(const Poly *p, int exp
在 C# 中是否没有字符串的 OR 操作数? 我正在查看 Microsoft C# 操作数页面 - 没有关于字符串的任何类型的 OR。 我有一个要写的 if 语句: if (Convert.ToStr
下面的函数左移一个double操作数: double shl(double x,unsigned long long n) { unsigned long long* p = (unsigne
我在 Linux 中使用了以下简单的 ksh 脚本 #!/bin/ksh set -x ### Process list of *.dat files if [ -f *.dat ] then pri
我有一个使用 Entity Framework 的查询。它有许多不同的操作数,我对其优先级感到困惑。我得到了错误的结果。我需要所有 IsPaid == true 或 IsPaid == null 的记
我有以下代码来尝试创建一个约束数组以添加到 View 中: let views = ["button": button] let metrics = ["margin": 16] var constr
这个问题在这里已经有了答案: How to compare one value against multiple values - Swift (8 个答案) 关闭 6 年前。 我有一种情况,我必须
我使用 jquery $.ajax 将请求发送到服务器,它返回 JSON。 $.ajax({ url: 'moreMonth.ajax', data: { startIndex: id },
我的问题是程序没有按照“他”的预期读取代码。 我有 if (hero.getPos() == (6 | 11 | 16)) { move = new Object[] {"Up", "Righ
我在对象中创建线程时遇到问题。错误是需要作为一元“&”操作数的左值 CPP文件 #include "AirQ.h" static int i=0; AirQ::AirQ(int pinNo, bool
我是一名优秀的程序员,十分优秀!