- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在开发一个简单的包装器模板类,它在调用特殊成员函数时进行记录。这些函数不能默认,因为它们执行额外的日志记录相关任务。
template <typename T>
struct logger {
logger(T const& value) : value_(value) { /*...log...*/ }
logger(T&& value) : value_(std::move(value)) { /*...log...*/ }
logger(logger const& other) : value_(other.value_) { /*...log...*/ }
logger(logger&& other) : value_(std::move(other.value_)) { /*...log...*/ }
T value_;
};
不幸的是,当包装类型是rvalue-reference时,copy-constructor 无法编译并显示以下错误消息:
error: cannot bind ‘int’ lvalue to ‘int&&’
原因是隐式复制构造函数对于右值引用成员的行为会有些不同:
[class.copy 12.8/15] The implicitly-defined copy/move constructor for a non-union class
X
performs a memberwise copy/move of its bases and members. [...] Letx
be either the parameter of the constructor or, for the move constructor, an xvalue referring to the parameter. Each base or non-static data member is copied/moved in the manner appropriate to its type:
- if the member is an array, each element is direct-initialized with the corresponding subobject of
x
;- if a member m has rvalue reference type
T&&
, it is direct-initialized withstatic_cast<T&&>(x.m)
;- otherwise, the base or member is direct-initialized with the corresponding base or member of
x
.
这让我想到了我的问题:如何编写一个通用的复制构造函数,它的行为就像一个隐式定义的复制构造函数,即使在使用时也是如此右值引用作为成员。
对于这种特殊情况,我可以为右值引用 添加额外的特化。但是,我正在寻找一种不限于单个成员且不引入代码重复的通用解决方案。
最佳答案
这里有龙。
logger(logger const& other) : value_(other.value_)
表达式other.value_
是 T const
类型的左值,例如int&
, int&&
或 int const
.
如果 T == int&&
, 你需要做一个 move
,因为表达式是左值。 move
相当于 static_cast<int&&>
, 所以你可以做 static_cast
也可以直接。
如果 T == int&
,不需要转换。
如果 T == int
,不需要转换。
对于定义为的复制构造函数:
logger(logger const& other) : value_(static_cast<T>(other.value_)) {/*...*/}
适用于第三种情况,这被定义为临时的引入,并可能导致额外的复制/移动,尽管我认为它可以并且将会被省略。
不依赖复制/移动省略的解决方案是引入 weird_cast
,在任何情况下都会产生所需的类型:
#include <type_traits>
template<class T, class U>
typename std::enable_if<std::is_reference<T>{}, T>::type
weird_cast(U& p)
{
return static_cast<T>(p);
}
template<class T, class U>
typename std::enable_if<not std::is_reference<T>{}, T const&>::type
weird_cast(U const& p)
{
return p;
}
int main()
{
int o = 42;
int & lo = o;
int && ro = std::move(o);
int const lco = o;
int&& r = weird_cast<int&&>(ro);
int& l = weird_cast<int& >(lo);
int d = weird_cast<int >(lco);
}
这类似于 std::forward
, 但也支持“转发”非引用类型。
龙在哪里?
[class.copy]/11 指定:
A defaulted copy/move constructor for a class
X
is defined as deleted ifX
has:
- [...]
- for the copy constructor, a non-static data member of rvalue reference type
- [...]
右值引用通常绑定(bind)到一个 xvalue 或 prvalue,即绑定(bind)到一个引用“接近其生命周期结束”的对象的表达式。由于生命周期不会通过函数边界得到延长,因此允许这样的“复制”很容易出错。
关于c++ - 具有右值引用成员的通用复制构造函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19040518/
我想知道最终用户按下了什么,所以我使用了 getch() 。 如果用户按右,我可以获得0xE0 0x4D。 如果用户按下Ctrl+右,我可以获得0xE0 0x47。 如果用户按下Shift+右,我可以
我已经构建了一个应用程序来搜索我的位置。 这是代码 var map; var gdir; var geocoder = null; var addressMarker; function init
我想为我的元素设计布局 View 。布局 View 在左 Angular 和右 Angular (同一行)有一个图像,将有 2 行单词,一行在第 1 行,另一行在第 2 行。我该如何实现? It
我有一个很长的线性(分支不多)流程图,在 graphviz 中显示为要么太高而无法放在单个页面上,要么太宽(如果方向是从左到右) 是否有一种简单的方法可以让 graphviz 以从左到右,然后向下,然
我一直摸不着头脑,但运气不好。设计器有一个包含 3 栏的站点、两个侧边栏和一个主要内容区域。 专为桌面设计,左栏、主要内容、右栏。但是,在较小的设备上,我们希望首先堆叠主要内容。 所以通常情况下,你可
我想要从上到下和从左到右组织的 css block 。 为了更好地解释这是一张图片,其中包含我到目前为止所获得的内容以及我希望使用 CSS 实现的内容: 代码如下: HTML: 1 2 3 4 5
当我问this question时,答案之一(现已删除)建议Either类型对应Curry-Howard correspondence中的XOR而不是OR,因为它不能同时是Left和Right。 真相
如果一行中六个观察值中至少有三个是 != NA,我想计算该行的平均值。如果存在四个或更多 NA,则平均值应显示为 NA。 给出平均值的例子,忽略了 NA: require(dplyr) a % mut
我有一个由 9 列组成的数据框,其中包含一个因素 list 。每行可以填充所有 9 列(因为在该行中包含 9 个“事物”),但大多数没有(大多数有 3-4 个)。列也不是特定的,就像第 1 列和第 3
这是我第一次尝试使用 R 构建函数。基本上我的预期目标如下。 使用 RoogleVision 包与 Google Cloud Vision API 通信 函数遍历目录中的图片 从每张图片的 Googl
使用: mean (x, trim=0.05) 从分布的每一侧移除 2.5%,这对于对称的双尾数据来说很好。但是如果我有一个尾部或高度不对称的数据,我希望能够只删除分布的一侧。有没有这个功能,还是我自
我想保留重复的列,并删除唯一的列。这些列将具有相同的值,但名称不同。 x1 = rnorm(1:10) x2 = rnorm(1:10) x3 = x1 x4 = rnorm(1:10) x5 = x
是否可以使WPF工具栏中的元素的Right水平对齐方式正确? 我尝试将内部元素添加到Grid中,并将ColumnDefinition分配给Left / Right。我
datatable(head(iris)) 如何将我的列居中,使其位于我的列名称的正下方? 最佳答案 您可以使用options 下的columnDefs 自变量。将 className 设置为 dt-
我是 R 的新手,但我正在尝试在 R 中制作滑动窗口。 使用循环我可以像这样,但这变得非常低效。 results=c(1:7) letters=c("A","B","C","D","E","F","G
假设我有这个 .txt 文件: here is line 1 here is line 2 here is line 3 here is line 4 我想将此字符串粘贴到第 3 行和第 4 行之间:
假设我有这个 .txt 文件: here is line 1 here is line 2 here is line 3 here is line 4 我想将此字符串粘贴到第 3 行和第 4 行之间:
我想知道我的环境中有什么类型的对象。 我可以像这样显示谁在那里: ls() 但是运行类似的东西 sapply(ls(), class) (显然)不会告诉我们我们拥有什么类型(类)的对象(函数、数字、因
我想创建一个带有水平标签的树状图,但让叶子根据它们的高度悬挂,而不是仅仅下降到图的边缘。 例子: par(mfrow = c(1,2)) hc <- hclust(dist(USArrests), "
我的 CSS 中有一个元素,如下所示 .xyz{ position:absolute; left:50%; } 现在正如预期的那样,当我减小浏览器窗口的宽度时,这个元素向左移动
我是一名优秀的程序员,十分优秀!