- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在编写五大类(复制构造函数、复制赋值运算符、移动构造函数、移动赋值运算符、析构函数)。我在复制构造函数语法方面遇到了一些障碍。
假设我有一个 foo 类,它具有以下私有(private)成员:
template<class data> // edit
class foo{
private:
int size, cursor; // Size is my array size, and cursor is the index I am currently pointing at
data * dataArray; // edit
}
如果我要为此编写一个任意大小的构造函数 X
它看起来像这样。
template<class data> // edit
foo<data>::foo(int X){
size = X;
dataArray = new data[size];
cursor = 0; // points to the first value
}
现在,如果我想为另一个名为 bar
的对象创建一个复制构造函数我需要做以下事情:
template<class data> // edit
foo<data>::foo(foo &bar){
foo = bar; // is this correct?
}
假设我有重载的 =
来自以下代码:
template<class data> // edit
foo<data>::operator=(foo &someObject){
if(this != someObject){
size = someObject.size;
cursor = someObject.cursor;
delete[] dataArray;
dataArray = new data[size];
for(cursor = 0; cursor<size-1;cursor++)
dataArray[cursor] = someObject.dataArray[cursor];
}
else
// does nothing because it is assigned to itself
return *this;
}
我的复制构造函数是否正确?或者应该foo = bar
而是 *this = bar
?
我对模板化构造函数还是个新手,所以如果我在代码中犯了任何错误,请告诉我,我会更正它。
编辑 1:感谢 Marcin 在下面提供的答案,我对上面的代码进行了一些编辑,使其在语法上更加正确,并用 //edit
对其进行了评论。它们总结在下面的列表中:
template<classname data>
,不正确的必须是 template <typename data>
或 template <class data>
分别用于函数和类。 int*dataArray;
这误用了模板,应该是 data* dataArray;
最佳答案
实现你想要的最好方法是使用一个已经处理分配、复制和移动的类,为你处理它的内存管理。 std::vector
正是这样做的,并且可以直接替换您动态分配的数组和大小。执行此操作的类通常称为 RAII类。
话虽如此,假设这是正确实现各种特殊成员函数的练习,我建议您通过 copy and swap idiom 继续. (有关更多详细信息和评论,请参见关于 SO 的 What is the copy and swap idiom?)。这个想法是根据复制构造函数定义赋值操作。
从成员、构造函数和析构函数开始。这些定义了您的类成员的所有权语义:
template <class data>
class foo {
public:
foo(const size_t n);
~foo();
private:
size_t size; // array size
size_t cursor; // current index
data* dataArray; // dynamically allocated array
};
template <class data>
foo<data>::foo(const size_t n)
: size(n), cursor(0), dataArray(new data[n])
{}
template <class data>
foo<data>::~foo() {
delete[] dataArray;
}
这里,内存在构造函数中分配,在析构函数中释放。接下来,编写复制构造函数。
template <class data>
foo<data>::foo(const foo<data>& other)
: size(other.size), cursor(other.cursor), dataArray(new data[other.size]) {
std::copy(other.dataArray, other.dataArray + size, dataArray);
}
(连同声明,foo(const foo& other);
在类主体内)。请注意它如何使用成员初始化列表将成员变量设置为 other
对象中的值。执行新的分配,然后在复制构造函数的主体中将数据从 other
对象复制到该对象中。
接下来是赋值运算符。您现有的实现必须执行大量指针操作,并且不是异常安全的。让我们看看如何更简单、更安全地完成此操作:
template <class data>
foo<data>& foo<data>::operator=(const foo<data>& rhs) {
foo tmp(rhs); // Invoke copy constructor to create temporary foo
// Swap our contents with the contents of the temporary foo:
using std::swap;
swap(size, tmp.size);
swap(cursor, tmp.cursor);
swap(dataArray, tmp.dataArray);
return *this;
}
(连同类内声明,foo& operator=(const foo& rhs);
)。
[-- 旁白:您可以通过按值接受函数参数来避免编写第一行(显式复制对象)。这是一回事,在某些情况下可能更有效:
template <class data>
foo<data>& foo<data>::operator=(foo<data> rhs) // Note pass by value!
{
// Swap our contents with the contents of the temporary foo:
using std::swap;
swap(size, rhs.size);
swap(cursor, rhs.cursor);
swap(dataArray, rhs.dataArray);
return *this;
}
但是,如果您还定义了移动赋值运算符,那么这样做可能会导致不明确的重载。 --]
它做的第一件事是创建一个被赋值对象的拷贝。这利用了复制构造函数,因此如何复制对象的细节只需要在复制构造函数中实现一次。
复制完成后,我们将内部结构与拷贝的内部结构进行交换。在函数体的末尾,tmp
拷贝超出范围,其析构函数清理内存。但这不是在函数开始时分配的内存;在我们与临时对象交换状态之前,它是我们的对象用来保存的内存。
通过这种方式,分配、复制和释放的细节都保留在它们所属的地方,在构造函数和析构函数中。赋值运算符只是简单地复制和交换。
除了更简单之外,这还有一个优点:它是异常安全的。在上面的代码中,分配错误可能导致在创建临时对象时抛出异常。但是我们还没有修改类的状态,所以即使赋值失败,我们的状态也保持一致(和正确)。
遵循相同的逻辑,移动操作变得微不足道。必须定义移动构造函数以简单地获取资源的所有权并将源(移出的对象)留在定义明确的状态中。这意味着将源的 dataArray
成员设置为 nullptr
,以便其析构函数中的后续 delete[]
不会导致问题。
移动赋值运算符的实现方式与复制赋值类似,但在这种情况下,不太关心异常安全性,因为您只是窃取了源对象的已分配内存。在完整的示例代码中,我选择简单地交换状态。
可以看到一个完整的、可编译和可运行的例子here .
关于c++ - 使用赋值重载语法复制构造函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37834156/
你能解释一下这个作业是如何完成的吗, 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
我是一名优秀的程序员,十分优秀!