17370845950

c++中的Copy-and-Swap惯用法是什么_c++实现强异常安全的赋值运算符
答案:Copy-and-Swap通过按值传参拷贝对象,再交换资源,确保异常安全并自动处理自我赋值与资源释放。1. 参数按值传递触发拷贝构造,失败则原对象不变;2. swap交换当前对象与副本的资源,应标记为noexcept;3. 函数返回时,副本析构释放旧资源。该法依赖正确实现的拷贝构造、析构和swap函数,符合强异常安全保证,代码简洁且兼容移动语义。

在C++中,Copy-and-Swap是一种实现赋值运算符的惯用法,主要用于确保强异常安全保证(Strong Exception Safety Guarantee),同时自动处理自我赋值和资源管理问题。它通过“拷贝-交换”两个步骤完成赋值操作。

基本原理

Copy-and-Swap的核心思想是:先创建目标对象的一个副本,然后将当前对象的内部资源与该副本进行交换。由于拷贝操作在修改原对象前完成,若拷贝过程中抛出异常,原对象仍保持原有状态,从而实现强异常安全。

这种技术通常配合一个swap非成员函数使用,且要求类正确实现了拷贝构造函数、析构函数和swap函数(遵循三大法则或五大法则)。

典型实现方式

假设我们有一个管理动态内存的类:

class MyClass {
private:
    int* data;
    size_t size;

public: // 构造函数 MyClass(size_t s = 0) : size(s), data(s ? new int[s]{} : nullptr) {}

// 拷贝构造函数
MyClass(const MyClass& other)
    : size(other.size), data(other.size ? new int[other.size] : nullptr)
{
    std::copy(other.data, other.data + other.size, data);
}

// 析构函数
~MyClass() { delete[] data; }

// 赋值运算符 - 使用 Copy-and-Swap
MyClass& operator=(MyClass rhs) {
    swap(*this, rhs);
    return *this;
}

// 友元 swap 函数(推荐)
friend void swap(MyClass& lhs, MyClass& rhs) noexcept {
    std::swap(lhs.data, rhs.data);
    std::swap(lhs.size, rhs.size);
}

};

关键点解析

  • 参数按值传递:operator= 的参数是按值传入的,这会自动触发拷贝构造函数,生成原始对象的一个副本。如果拷贝失败(如内存不足),异常会在赋值函数体执行前抛出,原对象未被修改。
  • swap交换资源:调用 swap 将当前对象的资源与副本交换。swap 应标记为 noexcept,避免在此阶段抛出异常。
  • 析构旧数据:当函数返回时,rhs 离开作用域,其析构函数自动释放交换前本对象的数据(即旧资源)。

为何能提供强异常安全

强异常安全意味着:操作要么完全成功,要么不改变对象状态。Copy-and-Swap满足这一点,因为:

  • 所有可能抛出异常的操作(拷贝构造)都在修改对象前完成。
  • swap 和析构调用通常不会抛出异常(swap 应设计为 noexcept)。

基本上就这些。只要正确实现拷贝构造、析构和 swap,Copy-and-Swap 能写出简洁、安全、自洽的赋值运算符,无需手动处理自我赋值或清理旧资源。在现代C++中,这一惯用法也自然兼容移动语义(结合 move constructor 后效率更高)。