- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
David Hollman 最近在推特上发布了以下示例(我稍微简化了):
struct FooBeforeBase {
double d;
bool b[4];
};
struct FooBefore : FooBeforeBase {
float value;
};
static_assert(sizeof(FooBefore) > 16);
//----------------------------------------------------
struct FooAfterBase {
protected:
double d;
public:
bool b[4];
};
struct FooAfter : FooAfterBase {
float value;
};
static_assert(sizeof(FooAfter) == 16);
您可以检查布局 in clang on godbolt并看到大小改变的原因是在 FooBefore
中,成员 value
放置在偏移量 16 处(保持与 FooBeforeBase
的完全对齐 8 >) 而在 FooAfter
中,成员 value
放置在偏移量 12 处(有效地使用 FooAfterBase
的尾部填充)。
我很清楚 FooBeforeBase
是标准布局,但 FooAfterBase
不是(因为它的非静态数据成员并不都具有相同的访问控制, [class.prop]/3)。但是,FooBeforeBase
的标准布局需要这种填充字节,这是怎么回事?
gcc 和 clang 都重用 FooAfterBase
的填充,以 sizeof(FooAfter) == 16
结束。但 MSVC 没有,以 24 结尾。是否有符合标准要求的布局,如果没有,为什么 gcc 和 clang 会这样做?
有一些困惑,所以只是为了澄清:
FooBeforeBase
是标准布局FooBefore
是 not(它和基类都有非静态数据成员,类似于 this example 中的 E
)FooAfterBase
是 not(它具有不同访问权限的非静态数据成员)FooAfter
是 not(出于上述两个原因)最佳答案
这个问题的答案不是来自标准,而是来自 Itanium ABI(这就是为什么 gcc 和 clang 有一种行为而 msvc 做其他事情的原因)。该 ABI 定义了 a layout ,就本问题而言,其相关部分是:
For purposes internal to the specification, we also specify:
- dsize(O): the data size of an object, which is the size of O without tail padding.
和
We ignore tail padding for PODs because an early version of the standard did not allow us to use it for anything else and because it sometimes permits faster copying of the type.
虚拟基类以外的成员的放置定义为:
Start at offset dsize(C), incremented if necessary for alignment to nvalign(D) for base classes or to align(D) for data members. Place D at this offset unless [... not relevant ...].
术语 POD 已从 C++ 标准中消失,但它意味着标准布局和可轻松复制。在这个问题中,FooBeforeBase
是一个 POD。 Itanium ABI 忽略尾部填充 - 因此 dsize(FooBeforeBase)
为 16。
但是 FooAfterBase
不是 POD(它可以简单地复制,但它是 not 标准布局)。结果,尾部填充没有被忽略,所以 dsize(FooAfterBase)
只是 12,而 float
可以直接到那里。
正如 Quuxplusone 在 related answer 中指出的那样,这会产生有趣的结果。 , 实现者通常还假设尾部填充没有被重用,这对这个例子造成了严重破坏:
#include <algorithm>
#include <stdio.h>
struct A {
int m_a;
};
struct B : A {
int m_b1;
char m_b2;
};
struct C : B {
short m_c;
};
int main() {
C c1 { 1, 2, 3, 4 };
B& b1 = c1;
B b2 { 5, 6, 7 };
printf("before operator=: %d\n", int(c1.m_c)); // 4
b1 = b2;
printf("after operator=: %d\n", int(c1.m_c)); // 4
printf("before std::copy: %d\n", int(c1.m_c)); // 4
std::copy(&b2, &b2 + 1, &b1);
printf("after std::copy: %d\n", int(c1.m_c)); // 64, or 0, or anything but 4
}
这里,=
做了正确的事(它不会覆盖 B
的尾部填充),但是 copy()
有一个库简化为 memmove()
的优化 - 它不关心尾部填充,因为它假设它不存在。
关于c++ - 标准布局和尾部填充,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53837373/
我最近在读 CSAPP。在 10.9 节中,它说标准 I/O 不应该与 socket 一起使用,原因如下: (1) The restrictions of standard I/O Restricti
似乎是一个足够标准的问题,可以保证解决方案中的标准设计: 假设我想在文件中写入 x+2(或更少)个字符串。 x 字符串构成一个部分的内容,这两个字符串构成该部分的页眉和页脚。要注意的是,如果内容中没有
代码版本管理 在项目中,代码的版本管理非常重要。每个需求版本的代码开发在版本控制里都应该经过以下几个步骤。 在master分支中拉取该需求版本的两个分支,一个feature分支,
我有以下sql查询,我需要获取相应的hibernate条件查询 SELECT COUNT(DISTINCT employee_id) FROM erp_hr_payment WHERE payment
所以我正在编写一些代码,并且最近遇到了实现一些 mixin 的需要。我的问题是,设计混音的正确方法是什么?我将使用下面的示例代码来说明我的确切查询。 class Projectile(Movable,
我的环境变量包含如下双引号: $echo $CONNECT_SASL_JAAS_CONFIG org.apache.kafka.common.security.plain.PlainLoginModu
示例: /** * This function will determine whether or not one string starts with another string. * @pa
有没有办法在 Grails 中做一个不区分大小写的 in 子句? 我有这个: "in"("name", filters.tags) 我希望它忽略大小写。我想我可以做一个 sqlRestriction
我搜索了很长时间,以查找将哪些boost库添加到std库中,但是我只找到了一个新库的完整列表(如此处:http://open-std.org/jtc1/sc22/wg21/docs/library_t
我已经通过使用这个肮脏的黑客解决了我的问题: ' Filter managerial functions ActiveSheet.Range("$A$1:$BW$2211").Auto
因此,我很难理解我需要遵循的标准,以便我的 Java 程序能够嵌入 HTML。我是否只需将我的主类扩展到 Applet 类,或者我还需要做更多的事情吗?另外,在我见过的每个 Applet 示例中,它都
我对在 Hibernate 中使用限制有疑问。 我必须创建条件,设置一些限制,然后选择日期字段最大值的记录: Criteria query = session.createCriteria(Stora
我有标准: ICriteria criteria = Session.CreateCriteria() .SetFetchMode("Entity1", FetchMo
我很难编写条件来选择所有子集合或孙集合为空的实体。我可以将这些作为单独的条件来执行,但我无法将其组合成一个条件。 类结构: public class Component { p
@Entity class A { @ManyToMany private List list; ... } @Entity class B { ... } 我想使用条件(不是 sql 查询)从 A
我的数据库中有以下表结构: Table A: Table B: Table C: _______________
请帮助我: 我有下一张 table : 单位 ID 姓名 用户 ID 姓名 利率 单位 ID 用户 ID 我不明白如何从 SQL 创建正确的条件结构: 代码: SELECT * FROM Unit W
我正在构建一个包含项目的网站,每个项目都有一个页面,例如: website.com/book/123 website.com/film/456 website.com/game/789 每个项目都可以
我需要使用两个属性的组合来过滤结果列表。一个简单的 SQL 语句如下所示: SELECT TOP 10 * FROM Person WHERE FirstName + ' ' + LastName L
我有一个“ super 实体”SuperEntity 和三个扩展父类(super class)的实体 ChildEntity1、...、ChildEntity3。 搜索数据库中的所有实体很容易,即我们
我是一名优秀的程序员,十分优秀!