- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我注意到 std::string
的(实际上是 std::basic_string
的)移动赋值运算符是 noexcept
。这对我来说很有意义。但后来我注意到没有一个标准容器(例如,std::vector
、std::deque
、std::list
、std::map
) 声明其移动赋值运算符 noexcept
。这对我来说意义不大。例如,std::vector
通常实现为三个指针,并且指针当然可以移动分配而不会引发异常。然后我认为问题可能在于移动容器的分配器,但是 std::string
也有分配器,所以如果这是问题,我希望它会影响 std::string
.
那么为什么std::string
的移动赋值运算符是noexcept
,而标准容器的移动赋值运算符却不是?
最佳答案
我相信我们正在研究一个标准缺陷。 noexcept
规范,如果要应用于移动赋值运算符,则有些复杂。无论我们谈论的是basic_string
,我都相信这个说法是正确的。或 vector
.
基于 [container.requirements.general]/p7 我对容器移动赋值运算符应该做的事情的英文翻译是:
C& operator=(C&& c)
If
alloc_traits::propagate_on_container_move_assignment::value
istrue
, dumps resources, move assigns allocators, and transfers resources fromc
.If
alloc_traits::propagate_on_container_move_assignment::value
isfalse
andget_allocator() == c.get_allocator()
, dumps resources, and transfers resources fromc
.If
alloc_traits::propagate_on_container_move_assignment::value
isfalse
andget_allocator() != c.get_allocator()
, move assigns eachc[i]
.
注意事项:
alloc_traits
指allocator_traits<allocator_type>
.
当alloc_traits::propagate_on_container_move_assignment::value
是 true
可以指定移动赋值运算符noexcept
因为它要做的就是释放当前资源,然后从源头窃取资源。同样在这种情况下,分配器也必须被移动分配,并且移动分配必须是 noexcept
容器的移动分配为 noexcept
.
当alloc_traits::propagate_on_container_move_assignment::value
是 false
,并且如果两个分配器相等,那么它将执行与#2 相同的操作。但是,直到运行时才知道分配器是否相等,因此您不能基于 noexcept
关于这种可能性。
当alloc_traits::propagate_on_container_move_assignment::value
是 false
,并且如果两个分配器不相等,则必须移动分配每个单独的元素。这可能涉及向目标添加容量或节点,因此本质上是 noexcept(false)
.
总之:
C& operator=(C&& c)
noexcept(
alloc_traits::propagate_on_container_move_assignment::value &&
is_nothrow_move_assignable<allocator_type>::value);
而且我看不到对 C::value_type
的依赖在上面的规范中,所以我相信它应该同样适用于 std::basic_string
尽管 C++11 另有说明。
更新
在下面的评论中,哥伦布正确地指出,事情一直在逐渐发生变化。我上面的评论是相对于 C++11 的。
对于 C++17 草案(在这一点上看起来很稳定),情况发生了一些变化:
如果 alloc_traits::propagate_on_container_move_assignment::value
是 true
,规范现在需要 allocator_type
的移动分配不抛出异常 (17.6.3.5 [allocator.requirements]/p4)。所以不再需要检查is_nothrow_move_assignable<allocator_type>::value
.
alloc_traits::is_always_equal
已添加。如果这是真的,那么可以在编译时确定上面第 3 点不能抛出,因为可以转移资源。
所以新的 noexcept
容器规范可以是:
C& operator=(C&& c)
noexcept(
alloc_traits::propagate_on_container_move_assignment{} ||
alloc_traits::is_always_equal{});
而且,对于 std::allocator<T>
, alloc_traits::propagate_on_container_move_assignment{}
和 alloc_traits::is_always_equal{}
都是真的。
现在也在 C++17 草案中,vector
和 string
移动赋值准确这个noexcept
规范。然而,其他容器带有此 noexcept
的变体规范。
如果您关心这个问题,最安全的做法是测试您关心的容器的显式特化。我已经为 container<T>
做到了这一点。对于 VS,libstdc++ 和 libc++ 在这里:
http://howardhinnant.github.io/container_summary.html
这项调查大约有一年的历史,但据我所知仍然有效。
关于c++ - 为什么容器移动赋值运算符不是 noexcept?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12332772/
Or 运算符 对两个表达式进行逻辑“或”运算。 result = expression1 Or expression2 参数 result 任意数值变量。 expression1 任意
Not 运算符 对表达式执行逻辑非运算。 result = Not expression 参数 result 任意数值变量。 expression 任意表达式。 说明 下表显示如何
Is 运算符 比较两个对象引用变量。 result = object1 Is object2 参数 result 任意数值变量。 object1 任意对象名。 object2 任意
\ 运算符 两个数相除并返回以整数形式表示的结果。 result = number1\number2 参数 result 任意数值变量。 number1 任意数值表达式。 numbe
And 运算符 对两个表达式进行逻辑“与”运算。 result = expression1 And expression2 参数 result 任意数值变量。 expression1
运算符(+) 计算两个数之和。 result = expression1 + expression2 参数 result 任意数值变量。 expression1 任意表达式。 exp
我对此感到困惑snippet : var n1 = 5-"4"; var n2 = 5+"4"; alert(n1); alert(n2); 我知道 n1 是 1。那是因为减号运算符会将字符串“4”转
我想我会得到 12,而不是 7。 w++,那么w就是4,也就是100,而w++, w 将是 8,1000;所以 w++|z++ 将是 100|1000 = 1100 将是 12。 我怎么了? int
Xor 运算符 对两个表达式进行逻辑“异或”运算。 result = expression1 Xor expression2 参数 result 任意数值变量。 expression1
Mod 运算符 两个数值相除并返回其余数。 result = number1 Mod number2 参数 result 任意数值变量。 number1 任意数值表达式。 numbe
Imp 运算符 对两个表达式进行逻辑蕴涵运算。 result = expression1 Imp expression2 参数 result 任意数值变量。 expression1 任
Eqv 运算符 执行两个表达式的逻辑等价运算。 result = expression1 Eqv expression2 参数 result 任意数值变量。 expression1 任
我有一个运算符重载的简单数学 vector 类。我想为我的运算符(operator)获取一些计时结果。我可以通过计时以下代码轻松计时我的 +=、-=、*= 和/=: Vector sum; for(s
我是用户定义比较运算符的新手。我正在读一本书,其中提到了以下示例: struct P { int x, y; bool operator、运算符<等),我们
在 SQL 的维基百科页面上,有一些关于 SQL 中 bool 逻辑的真值表。 [1] 维基百科页面似乎来源于 SQL:2003 标准。 等号运算符 (=) 的真值表与 SQL:2003 草案中的 I
我遇到了一个奇怪的 C++ 运算符。 http://www.terralib.org/html/v410/classoracle_1_1occi_1_1_number.html#a0f2780081f
我正在阅读关于 SO 和 answers 中的一个问题,它被提到为: If no unambiguous matching deallocation function can be found, pr
我偶然发现了这个解决方案,但我无法理解其中到底发生了什么。谁能解释一下! 据我了解,它试图通过计算一半的单元格然后将其加倍来计算 a*b 网格中的单元格数量。但是我无法理解递归调用。 请不要建议其他解
Go的基本类型 布尔类型bool 长度:1字节 取值:布尔类型的取值只能是true或者false,不能用数字来表示 整型 通用整型 int / uint(有符号 / 无符号,下面也类似) 长度:根据运
在本教程中,您将学习JavaScript中可用的不同运算符,以及在示例的帮助下如何使用它们。 什么是运算符? 在JavaScript中,运算符是一种特殊符号,用于对运算数(值和变量)执行操作。例如,
我是一名优秀的程序员,十分优秀!