- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
考虑以下片段(假设 T
是平凡可构造和平凡可破坏的):
std::optional<T> opt;
opt.emplace();
T& ref = opt.value();
opt.emplace();
// is ref guaranteed to be valid here?
来自 the definition of std::optional
我们知道包含的实例保证在 std::optional
容器内分配,因此我们知道引用 ref
将始终引用相同的内存位置。是否存在指向对象被销毁并再次构造后该引用将不再有效的情况?
最佳答案
C++20 有如下规则,[basic.life]/8:
If, after the lifetime of an object has ended and before the storage which the object occupied is reused or released, a new object is created at the storage location which the original object occupied, a pointer that pointed to the original object, a reference that referred to the original object, or the name of the original object will automatically refer to the new object and, once the lifetime of the new object has started, can be used to manipulate the new object, if the original object is transparently replaceable (see below) by the new object. An object o1 is transparently replaceable by an object o2 if:
- the storage that o2 occupies exactly overlays the storage that o1 occupied, and
- o1 and o2 are of the same type (ignoring the top-level cv-qualifiers), and
- o1 is not a complete const object, and
- neither o1 nor o2 is a potentially-overlapping subobject (6.7.2), and
- either o1 and o2 are both complete objects, or o1 and o2 are direct subobjects of objects p1 and p2 ,respectively, and p1 is transparently replaceable by p2.
这表明只要T
不是 const 限定的,破坏了 T
在 std::optional<T>
里面然后重新创建它应该导致对旧对象的引用自动引用新对象。正如评论部分所指出的,这是对旧行为的更改,取消了 T
的要求。不得包含 const 限定或引用类型的非静态数据成员。 (编辑:我之前断言更改是追溯性的,因为我将它与 C++20 中的不同更改混淆了。我不确定 N4858 中指示的 RU 007 和 US 042 的决议是否具有追溯性,但我怀疑答案是肯定的,因为需要更改来修复涉及标准库模板的代码,这些模板可能不是打算从 C++11 到 C++17 被破坏。)
但是,我们假设新的 T
“在 [旧] 对象占用的存储被重用或释放之前”创建对象。如果我正在编写标准库的“对抗性”实现,我可以将其设置为 emplace
调用在创建新的 T
之前重用底层存储目的。这将防止旧的 T
对象被新对象透明地替换。
实现如何“重用”存储?通常,底层存储可以这样声明:
union {
char no_object;
T object;
};
当 optional
的默认构造函数时被称为,no_object
已初始化(值无关紧要)1。一个emplace()
调用检查是否有 T
是否反对(通过检查此处未显示的标志)。如果 T
对象存在,则 object.~T()
叫做。最后,类似于 construct_at(addressof(object))
的内容被调用以构建新的 T
对象。
并不是说任何实现都会这样做,但您可以想象一个实现,在对 object.~T()
的调用之间和 construct_at(addressof(object))
, 重新初始化 no_object
成员。这将是对先前由 object
占用的存储空间的“重用”。 .这意味着不满足 [basic.life]/8 的要求。
当然,您的问题的实际答案是:(1) 没有理由让实现做这样的事情,并且 (2) 即使实现做了,开发人员也会确保您的代码仍然运行好像 T
对象被透明地替换。在标准库实现合理的假设下,您的代码是合理的,并且编译器开发人员不喜欢使用该属性破坏代码,因为这样做会不必要地激怒他们的用户。
但是,如果编译器开发人员倾向于破坏您的代码(基于未定义行为越多,编译器可以优化的越多的论点),那么他们甚至可以破坏您的代码,甚至 <optional>
头文件。用户需要将标准库视为一个“黑匣子”,它只保证标准明确保证的内容。因此,根据对标准的迂腐阅读,未指定是否尝试访问 ref
。在第二个之后emplace
调用具有未定义的行为。如果未指定它是否是 UB,则允许编译器在需要时开始将其视为 UB。
1 这是历史原因; C++17 要求 constexpr
构造函数只初始化 union 的一个变体成员。此规则在 C++20 中被废除,因此 C++20 实现可以省略 no_object
成员(member)。
关于c++ - std::optional<>::emplace() 是否会使对内部值的引用无效?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72349969/
我有一个接受以下参数的函数: int setvalue(void (*)(void *)); 为了满足参数:void (*)(void *),我创建了这样一个函数: static void *
我有以下代码: typedef void VOID; int f(void); int g(VOID); 在 C 中编译得很好(在 Fedora 10 上使用 gcc 4.3.2)。与 C++ 编译的
这个问题已经有答案了: Is f(void) deprecated in modern C and C++? [duplicate] (6 个回答) 已关闭 7 年前。 B.A.T.M.A.N./A.
我在 ASP.NET Core 3.1 项目上有以下 Identity Server 4 配置: services .AddIdentityServer(y => { y.Events.R
我们有一个 O365 租户,一切都是开箱即用的。租户放置在德国云中,而不是全局 (office.de) 中。我们还开发了一个 Office 插件,使用 OAuth 2.0 授权访问共享点。首先,我们向
我有一个如下所示的路由 routes.MapRoute( name: "Default", url: "{controller}/{action}/{i
我正在尝试使用 OAuth2.0 访问 google 文档。我已经从 Google API 控制台获取了客户端 ID 和 key 。但是当我运行这段代码时,我收到了异常。如果我遗漏了什么,有人可以建议
此代码有效: let mut b: Vec = Vec::with_capacity(a.len()); for val in a.iter() { b.push(val); } 此代码不起作
使用 client_credintials 授权类型请求 EWS oauth2 v2.0 的访问 token 时出现错误。 https://login.microsoftonline.com/tena
我通过 Java 应用程序使用 Google 电子表格时遇到了问题。我创建了应用程序,该应用程序运行了 1 年多,没有任何问题,我什至在 Create Spreadsheet using Google
如何创建 匹配所有无效 Base64 字符的正则表达式?我在堆栈上找到了 [^a-zA-Z0-9+/=\n\r].*$ 但是当我尝试时我得到了带有 - 符号的结果字符串.我根本不知道正则表达式,任何人
我从 Gitlab CI/CD Pipelines 获得错误信息:yaml invalid。问题是由 .gitlab-ci.yml 脚本的第五行引起的: - 'ssh deployer@gita
我有 3 个数据源,设置如下: @Configuration @Component public class DataSourceConfig { @Bean("foo") @Conf
你好,我想用bulkCreate ex 插入数据: [ { "typeId": 5, "devEui": "0094E796CBFCFEF9", "application_name": "Pressu
UIApplicationExitsOnSuspend 不会强制我的应用程序退出。我已经清理过目标、删除了应用程序、重建并重新安装了很多次。 我确实需要退出我的应用程序。 最佳答案 您是否链接了 SD
在 iPhone 配置门户上,显示我的 iPhone 团队配置配置文件无效。有一个“由 Xcode 管理”文本。 “续订”按钮被禁用。 我该如何解决这个问题?谢谢 最佳答案 使用 Xcode 3.2.
好的,所以今天我用我们的“实时”数据库中的新信息更新了我的数据库……从那时起,我的一个表格就出现了问题。如果您需要任何代码,请告诉我,我将对其进行编辑并发布所需的代码... 我有一个报告表格,其中有一
我有一个结构体,其中有一个元素表示为 void (*func)(); 我知道 void 指针通常用于函数指针,但我似乎无法定义该函数。我不断收到取消引用指向不完整类型的指针。我用谷歌搜索了一下但没有结
我正在尝试使用 Coldfusion 9 从 ning 网络获取凭证,所以首先这是测试 api 的 curl 语法: curl -k https://external.ningapis.com/xn/
这个问题已经有答案了: Does C have references? (2 个回答) 已关闭 4 年前。 我正在学习 C 语言引用,这是我的代码: #include int main(void)
我是一名优秀的程序员,十分优秀!