- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
所以玩转 move 语义。
所以我第一次看这个是这样的:
class String
{
char* data;
int len;
public:
// Normal rule of three applied up here.
void swap(String& rhs) throw()
{
std::swap(data, rhs.data);
std::swap(len, rhs.len);
}
String& operator=(String rhs) // Standard Copy and swap.
{
rhs.swap(*this);
return *this;
}
// New Stuff here.
// Move constructor
String(String&& cpy) throw() // ignore old throw construct for now.
: data(NULL)
, len(0)
{
cpy.swap(*this);
}
String& operator=(String&& rhs) throw()
{
rhs.swap(*this);
return *this;
}
};
看着这个。我认为根据 Move 赋值定义 Move 构造函数可能是值得的。它有一个很好的对称性,我喜欢它,因为它看起来也很干(并且像 copy-and-swap )。
所以我将 Move Constructor 重写为:
String(String&& cpy) throw()
: data(NULL)
, len(0)
{
operator=(std::move(cpy));
}
但这会产生歧义错误:
String.cpp:45:9: error: call to member function 'operator=' is ambiguous
operator=(std::move(rhs));
^~~~~~~~~
String.cpp:32:13: note: candidate function
String& operator=(String rhs)
^
String.cpp:49:13: note: candidate function
String& operator=(String&& rhs) throw()
^
1 error generated.
因为我在传递参数时使用了 std::move()
,所以我希望它能绑定(bind)到 Move 赋值运算符。我做错了什么?
最佳答案
What am I doing wrong?
尝试根据另一个来编写一个特殊成员函数的情况应该很少见。每个特殊成员通常都需要特别注意。 如果在使每个特殊成员尽可能高效的练习之后,您看到了合并代码的机会,那么,也只有在那时,才开始努力。
以在特殊成员之间整合代码为目标是错误的起点。
第 1 步。开始尝试使用 = default
编写您的特殊成员。
第2步,如果失败,则自定义每一个不能写的=default
。
第 3 步。编写测试以确认第 2 步是否有效。
第 4 步。完成第 3 步后,查看是否可以在不牺牲性能的情况下进行代码整合。这可能涉及编写性能测试。
直接跳到第 4 步很容易出错,并且通常会导致严重的性能损失。
这是您示例的第 2 步:
#include <algorithm>
class String
{
char* data;
int len;
public:
String() noexcept
: data(nullptr)
, len(0)
{}
~String()
{
delete [] data;
}
String(const String& cpy)
: data(new char [cpy.len])
, len(cpy.len)
{
std::copy(cpy.data, cpy.data+cpy.len, data);
}
String(String&& cpy) noexcept
: data(cpy.data)
, len(cpy.len)
{
cpy.data = nullptr;
cpy.len = 0;
}
String& operator=(const String& rhs)
{
if (this != &rhs)
{
if (len != rhs.len)
{
delete [] data;
data = nullptr;
len = 0;
data = new char[rhs.len];
len = rhs.len;
}
std::copy(rhs.data, rhs.data+rhs.len, data);
}
return *this;
}
String& operator=(String&& rhs) noexcept
{
delete [] data;
data = nullptr;
len = 0;
data = rhs.data;
len = rhs.len;
rhs.data = nullptr;
rhs.len = 0;
return *this;
}
void swap(String& rhs) noexcept
{
std::swap(data, rhs.data);
std::swap(len, rhs.len);
}
};
更新
应该注意的是,在 C++98/03 中,不能成功重载参数仅在按值和按引用之间不同的函数。例如:
void f(int);
void f(int&);
int
main()
{
int i = 0;
f(i);
}
test.cpp:8:5: error: call to 'f' is ambiguous
f(i);
^
test.cpp:1:6: note: candidate function
void f(int);
^
test.cpp:2:6: note: candidate function
void f(int&);
^
1 error generated.
添加 const
没有帮助:
void f(int);
void f(const int&);
int
main()
{
f(0);
}
test.cpp:7:5: error: call to 'f' is ambiguous
f(0);
^
test.cpp:1:6: note: candidate function
void f(int);
^
test.cpp:2:6: note: candidate function
void f(const int&);
^
1 error generated.
这些相同的规则适用于 C++11,并且在不修改右值引用的情况下进行了扩展:
void f(int);
void f(int&&);
int
main()
{
f(0);
}
test.cpp:7:5: error: call to 'f' is ambiguous
f(0);
^
test.cpp:1:6: note: candidate function
void f(int);
^
test.cpp:2:6: note: candidate function
void f(int&&);
^
1 error generated.
因此,给出以下结果也就不足为奇了:
String& operator=(String rhs);
String& operator=(String&& rhs) throw();
结果是:
String.cpp:45:9: error: call to member function 'operator=' is ambiguous
operator=(std::move(rhs));
^~~~~~~~~
String.cpp:32:13: note: candidate function
String& operator=(String rhs)
^
String.cpp:49:13: note: candidate function
String& operator=(String&& rhs) throw()
^
1 error generated.
关于c++ - 尝试根据 move 赋值编写 move 构造函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19826045/
你能解释一下这个作业是如何完成的吗, var fe, f = document.forms[0], h; 哪个等于哪个。 最佳答案 以上等同于 var fe; var f = document.for
据我测试,这两种方法都有效,但我不知道哪一种最好,也不知道它们之间的区别,这就是我想知道的。 以下是两种方法: window.location = 'http://www.google.com'; w
我正在处理用字符串填充的 numpy 数组。我的目标是分配给第一个数组 a 的切片,值包含在较小尺寸的第二个数组 b 中。 我想到的实现如下: import numpy as np a = np.em
在我使用过的其他语言(如 Erlang 和 Python)中,如果我正在拆分字符串并且不关心其中一个字段,我可以使用下划线占位符。我在 Perl 中试过这个: (_,$id) = split('
我认为这似乎很简单,但我对调用、应用、绑定(bind)感到困惑。等等 我有一个事件监听器 red.addEventListener("click", function() { j = 0;
这个问题在这里已经有了答案: What is the python "with" statement designed for? (11 个答案) 关闭 7 年前。 使用有什么区别: iFile =
这个问题在这里已经有了答案: What is the python "with" statement designed for? (11 个答案) 关闭 7 年前。 使用有什么区别: iFile =
几周前我们开始写一篇关于 Haskell 的论文,刚刚接到我们的第一个任务。我知道 SO 不喜欢家庭作业问题,所以我不会问怎么做。相反,如果有人能将我推向正确的方向,我将不胜感激。鉴于它可能不是一个特
我正在尝试为我的函数的变量根分配一个值,但似乎不起作用。我不明白这个问题。 hw7.c:155:7:警告:赋值使指针来自整数而不进行强制转换[默认启用] root = 负载(&fp, 大小); 此代码
我昨天花了大约 5 个小时来完成这个工作,并使用这个网站的帮助让代码可以工作,但我认为我这样做的方式是一种作弊方式,我使用了 scanf 命令。无论如何,我想以正确的方式解决这个问题。多谢你们!哦,代
我需要一些帮助来解决问题。 我有这个文本文件: 我将文本内容输入到字符串二维数组中,并将其转换为整数二维数组。当我转换为 int 数组时,nan 被替换为零。现在,我继续查找二维数组中每行的最大值和最
假设我有一个只能移动的类型。我们停止现有的默认提供的构造函数,但 Rvalue 引用引入了一种新的“ flavor ”,我们可以将其用于签名的移动版本: class CantCopyMe { priv
假设我有两个简单的对象,我想创建第三个对象来连接它们的属性。这非常有效: (()=>{ const a1 = {a: 2, b: 3} const b1 = {a: 100, c: 5}
我想知道我是否可以稍后在这样的代码中为 VAR 赋值 var myView: UIView func createView() { myView = UIView() { let _view =
我遇到了一些 Javascript/HTML/CSS 代码的问题。我对创建网站还很陌生,所以请多多包涵。 我最终想做的是从 javascript 中提取一个动态值并使用它对一些 div(在容器中)进行
#include class Box{ public: int x; Box(){ x=0; std::cout No move construction thanks to RV
我发现在javascript中&=运算符是按位赋值: var test=true; test&=true; //here test is an int variable javascript中是否存在
请帮助完成赋值重载函数的执行。 这是指令: 赋值运算符 (=),它将源字符串复制到目标字符串中。请注意,目标的大小需要调整为与源相同。 加法 (+) 和赋值 (=) 运算符都需要能够进行级联运算。这意
我有一个名为 SortedArrayList 的自定义结构它根据比较器对其元素进行排序,我想防止使用 operator[] 进行分配. 示例: 数组列表.h template class Array
我是 python 的新手,我看到了这种为列表赋值的形式 color= ['red' if v == 0 else 'green' for v in y] 但是如果我尝试用 3 个数字来做,例如 co
我是一名优秀的程序员,十分优秀!