从拷贝到移动:C++ 移动构造与移动赋值是怎么被逼出来的?(附完整示例)

很多人在学习 C++ 时,第一次看到下面这两个函数会一脸懵:

cpp 复制代码
Box(Box&& other);            // 移动构造
Box& operator=(Box&& other); // 移动赋值

这两个函数看起来像"高级语法",

但实际上它们不是凭空出现的,而是被 性能问题 + 临时对象浪费 一步步逼出来的进化结果。

本文从"问题"出发,再用一个完整可运行的例子把它讲透。

一、最初阶段:只有拷贝构造

最早 C++ 只有拷贝语义:

cpp 复制代码
Box(const Box& other);

含义就是:

cpp 复制代码
复制一份资源

示例:

cpp 复制代码
Box a(10);
Box b = a;   // 拷贝构造

如果资源很小,问题不大。

但当成员是:

  • std::string
  • std::vector
  • 大数组
  • 文件句柄
  • 网络连接

复制就会变得 慢 + 占内存

二、问题升级:临时对象的浪费

看一个常见函数:

cpp 复制代码
Box createBox() {
    Box b(10);
    return b;
}

调用:

cpp 复制代码
Box a = createBox();

如果只有拷贝构造,流程是:

cpp 复制代码
构造 b
拷贝构造 a
析构 b

问题在于:

b 马上要死了,还完整复制一份资源,纯浪费。

于是一个需求出现了:

能不能不复制,而是"搬走资源"?

三、移动语义诞生 ------ "搬家"而不是"复印"

移动语义的核心思想:

如果对象马上要被销毁,就不要复制资源,直接转移所有权。

这就引出了 移动构造移动赋值

四、完整示例代码(核心)

下面是一个完整可运行的类,包含:

  • 构造
  • 拷贝构造
  • 移动构造
  • 拷贝赋值
  • 移动赋值
  • 析构
cpp 复制代码
#include <iostream>
using namespace std;

class Box {
public:
    int* data;

    // 构造
    Box(int v) {
        data = new int(v);
        cout << "构造\n";
    }

    // 拷贝构造
    Box(const Box& other) {
        data = new int(*other.data);
        cout << "拷贝构造\n";
    }

    // 移动构造
    Box(Box&& other) {
        data = other.data;       // 接管资源
        other.data = nullptr;    // 清空对方
        cout << "移动构造\n";
    }

    // 拷贝赋值
    Box& operator=(const Box& other) {
        cout << "拷贝赋值\n";
        if (this == &other) return *this;

        delete data;
        data = new int(*other.data);
        return *this;
    }

    // 移动赋值
    Box& operator=(Box&& other) {
        cout << "移动赋值\n";
        if (this == &other) return *this;

        delete data;           // 释放旧资源
        data = other.data;     // 接管资源
        other.data = nullptr;  // 清空对方
        return *this;
    }

    ~Box() {
        delete data;
        cout << "析构\n";
    }
};

Box createBox() {
    Box b(10);
    return b;
}

int main() {
    cout << "=== 移动构造示例 ===\n";
    Box a = createBox();   // 触发移动构造(或RVO)

    cout << "=== 移动赋值示例 ===\n";
    Box b(1);
    b = createBox();       // 触发移动赋值
}

五、可能看到的输出

不同编译器略有差异,常见输出:

cpp 复制代码
=== 移动构造示例 ===
构造
移动构造
析构
析构

=== 移动赋值示例 ===
构造
构造
移动赋值
析构
析构

有时你只会看到:

cpp 复制代码
构造

那是 RVO(返回值优化) 在工作 ------

编译器直接在目标位置构造对象,连移动都省掉了。

六、移动构造 vs 移动赋值区别

类型 场景 是否已有资源 核心动作
移动构造 新对象创建 没有 直接接管资源
移动赋值 对象已存在 先释放再接管

七、为什么要把对方清空?

cpp 复制代码
other.data = nullptr;

原因:

cpp 复制代码
不清空 → 析构 double free
清空 → delete nullptr 安全

八、现实类比

cpp 复制代码
拷贝构造 = 复印整箱书
移动构造 = 把箱子搬走
拷贝赋值 = 先扔旧书再复印新书
移动赋值 = 先扔旧书再搬新箱子

九、终极锚点总结

cpp 复制代码
拷贝构造 → 复制资源
移动构造 → 搬资源给新对象
拷贝赋值 → 复制资源给已有对象
移动赋值 → 清空自己再搬资源

移动语义不是复杂语法,

它是 "拷贝太贵"被逼出来的性能进化。

当你理解:

cpp 复制代码
资源复制 vs 资源转移

移动构造和移动赋值就再也不会混了。

相关推荐
古译汉书1 小时前
部分.exe文件打开但是一直显示界面,点击任务栏持续无反应
开发语言·单片机·嵌入式硬件
2301_817497331 小时前
C++中的装饰器模式高级应用
开发语言·c++·算法
m0_549416662 小时前
C++编译期字符串处理
开发语言·c++·算法
m0_581124192 小时前
C++中的适配器模式实战
开发语言·c++·算法
Coding茶水间2 小时前
基于深度学习的狗品种检测系统演示与介绍(YOLOv12/v11/v8/v5模型+Pyqt5界面+训练代码+数据集)
开发语言·人工智能·深度学习·yolo·目标检测·机器学习
£漫步 云端彡2 小时前
Golang学习历程【第十篇 方法(method)与接收者】
开发语言·学习·golang
u0109272712 小时前
C++与人工智能框架
开发语言·c++·算法
EmbedLinX2 小时前
嵌入式Linux C++常用设计模式
linux·c++·设计模式
挖矿大亨2 小时前
C++中空指针访问成员函数
开发语言·c++