C++11 引入了移动语义,它允许资源(如内存、文件句柄等)从一个对象转移到另一个对象,而不是像传统的拷贝语义那样创建资源的副本。移动语义通过引入右值引用、std::move
函数和移动构造函数等特性来实现。
虽然 C++ 标准库中的移动语义已经固定,但你可以在你的代码中实现自定义的移动语义。例如,你可以为你的类提供自定义的移动构造函数和移动赋值运算符,以便在需要时实现资源的转移。
这里有一个简单的例子,展示了如何为自定义类实现移动语义:
#include <iostream>
#include <string>
class MyString {
public:
// 默认构造函数
MyString() : data(nullptr), size(0) {}
// 拷贝构造函数
MyString(const MyString& other) {
data = new char[other.size + 1];
std::copy(other.data, other.data + other.size, data);
size = other.size;
}
// 移动构造函数
MyString(MyString&& other) noexcept : data(other.data), size(other.size) {
other.data = nullptr;
other.size = 0;
}
// 拷贝赋值运算符
MyString& operator=(const MyString& other) {
if (this != &other) {
delete[] data;
data = new char[other.size + 1];
std::copy(other.data, other.data + other.size, data);
size = other.size;
}
return *this;
}
// 移动赋值运算符
MyString& operator=(MyString&& other) noexcept {
if (this != &other) {
delete[] data;
data = other.data;
size = other.size;
other.data = nullptr;
other.size = 0;
}
return *this;
}
// 析构函数
~MyString() {
delete[] data;
}
// 获取字符串长度
size_t length() const {
return size;
}
// 获取字符串数据
const char* c_str() const {
return data;
}
private:
char* data;
size_t size;
};
int main() {
MyString s1("hello");
MyString s2 = std::move(s1); // 使用移动语义将 s1 的资源转移到 s2
std::cout << "s1 length: " << s1.length() << std::endl; // 输出 0,因为 s1 的资源已被移动
std::cout << "s2 length: " << s2.length() << std::endl; // 输出 5,因为 s2 的资源是从 s1 移动过来的
return 0;
}
在这个例子中,我们为 MyString
类提供了自定义的移动构造函数和移动赋值运算符,以便在需要时实现资源的转移。这样,当我们将一个 MyString
对象移动到另一个对象时,资源会被有效地转移,而不是被复制。这可以提高性能,特别是在处理大型资源时。