0

0

C++类的拷贝赋值运算符重载

P粉602998670

P粉602998670

发布时间:2025-09-18 08:02:01

|

367人浏览过

|

来源于php中文网

原创

c++类的拷贝赋值运算符重载

拷贝赋值运算符重载,简单来说,就是让你能用

=
给一个已经存在的C++对象赋值。它和拷贝构造函数不太一样,拷贝构造函数是用来创建一个新的对象,而拷贝赋值是修改一个已有的对象。

拷贝赋值运算符重载

想要搞定拷贝赋值运算符重载,主要得注意这几点:自赋值的处理、释放旧资源、分配新资源、以及返回对象的引用。

为什么要重载拷贝赋值运算符?

默认情况下,C++会提供一个拷贝赋值运算符,它会简单地将源对象的所有成员变量复制到目标对象。但如果你的类里有动态分配的内存,比如用

new
分配的,那默认的拷贝赋值运算符就只会复制指针,而不会复制指针指向的内容,这就是浅拷贝。浅拷贝会导致多个对象指向同一块内存,当其中一个对象释放了这块内存,其他的对象就变成了悬挂指针,访问就会出错。所以,当你的类里有动态分配的内存或者其他需要特殊处理的资源时,就需要重载拷贝赋值运算符,实现深拷贝,确保每个对象都有自己独立的资源。

立即学习C++免费学习笔记(深入)”;

如何编写一个安全的拷贝赋值运算符?

一个安全的拷贝赋值运算符需要考虑以下几个方面:

  1. 自赋值检查:首先要检查是不是自己给自己赋值,如果是,直接返回

    *this
    ,避免不必要的资源释放和分配。

  2. 释放旧资源:在赋值之前,先释放目标对象已经拥有的资源,防止内存泄漏。

  3. 分配新资源并复制数据:分配足够的内存,然后将源对象的数据复制到目标对象。

  4. 异常安全:确保在分配资源或复制数据时发生异常,对象的状态仍然是有效的,不会出现资源泄漏或数据损坏。一种常见的做法是先创建一个临时对象,将源对象的数据复制到临时对象,然后再将临时对象的数据交换到目标对象。

    云从科技AI开放平台
    云从科技AI开放平台

    云从AI开放平台

    下载
  5. 返回对象的引用:返回

    *this
    的引用,这样可以支持链式赋值,比如
    a = b = c;

下面是一个示例代码:

#include <iostream>
#include <cstring>

class MyString {
private:
    char* data;
    size_t length;

public:
    // 构造函数
    MyString(const char* str = nullptr) : data(nullptr), length(0) {
        if (str) {
            length = std::strlen(str);
            data = new char[length + 1];
            std::strcpy(data, str);
        }
    }

    // 拷贝构造函数
    MyString(const MyString& other) : data(nullptr), length(0) {
        length = other.length;
        data = new char[length + 1];
        std::strcpy(data, other.data);
    }

    // 拷贝赋值运算符
    MyString& operator=(const MyString& other) {
        // 自赋值检查
        if (this == &other) {
            return *this;
        }

        // 释放旧资源
        delete[] data;
        data = nullptr; // 重要:避免悬挂指针

        // 分配新资源并复制数据
        length = other.length;
        data = new char[length + 1];
        std::strcpy(data, other.data);

        // 返回对象的引用
        return *this;
    }

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

    // 打印字符串
    void print() const {
        if (data) {
            std::cout << data << std::endl;
        } else {
            std::cout << "(null)" << std::endl;
        }
    }
};

int main() {
    MyString str1("Hello");
    MyString str2("World");
    str2 = str1; // 使用拷贝赋值运算符
    str1.print(); // 输出 "Hello"
    str2.print(); // 输出 "Hello"

    return 0;
}

拷贝赋值运算符和移动赋值运算符有什么区别?

拷贝赋值运算符和移动赋值运算符都是用来给对象赋值的,但它们的应用场景和实现方式有所不同。拷贝赋值运算符用于源对象仍然需要保持有效的情况,它会复制源对象的数据到目标对象。而移动赋值运算符用于源对象不再需要保持有效的情况,它会将源对象的资源“移动”到目标对象,避免了不必要的复制,提高了效率。移动赋值运算符通常用于处理临时对象或者即将销毁的对象。移动赋值运算符的参数是一个右值引用,用

&&
表示。

如何处理拷贝赋值运算符中的异常?

在拷贝赋值运算符中,如果分配内存失败或者复制数据时发生异常,可能会导致资源泄漏或者数据损坏。为了保证异常安全,可以使用以下几种方法:

  1. 先分配再释放:先分配新的资源,然后再释放旧的资源。如果在分配新资源时发生异常,旧的资源仍然是有效的。

  2. 使用

    std::unique_ptr
    std::shared_ptr
    :这些智能指针可以自动管理资源,即使发生异常,也能保证资源被正确释放。

  3. Copy-and-Swap:创建一个临时对象,将源对象的数据复制到临时对象,然后将临时对象的数据和目标对象的数据进行交换。这种方法可以保证即使在复制数据时发生异常,目标对象的状态仍然是有效的。

下面是一个使用Copy-and-Swap的示例代码:

#include <iostream>
#include <cstring>
#include <algorithm>

class MyString {
private:
    char* data;
    size_t length;

public:
    // 构造函数
    MyString(const char* str = nullptr) : data(nullptr), length(0) {
        if (str) {
            length = std::strlen(str);
            data = new char[length + 1];
            std::strcpy(data, str);
        }
    }

    // 拷贝构造函数
    MyString(const MyString& other) : data(nullptr), length(0) {
        length = other.length;
        data = new char[length + 1];
        std::strcpy(data, other.data);
    }

    // 移动构造函数
    MyString(MyString&& other) noexcept : data(other.data), length(other.length) {
        other.data = nullptr;
        other.length = 0;
    }

    // 拷贝赋值运算符
    MyString& operator=(const MyString& other) {
        MyString temp(other); // 创建临时对象
        std::swap(data, temp.data); // 交换数据
        std::swap(length, temp.length);
        return *this;
    }

    // 移动赋值运算符
    MyString& operator=(MyString&& other) noexcept {
        if (this != &other) {
            delete[] data;
            data = other.data;
            length = other.length;
            other.data = nullptr;
            other.length = 0;
        }
        return *this;
    }

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

    // 打印字符串
    void print() const {
        if (data) {
            std::cout << data << std::endl;
        } else {
            std::cout << "(null)" << std::endl;
        }
    }
};

int main() {
    MyString str1("Hello");
    MyString str2("World");
    str2 = str1; // 使用拷贝赋值运算符
    str1.print(); // 输出 "Hello"
    str2.print(); // 输出 "Hello"

    MyString str3("Move");
    str3 = std::move(str1); // 使用移动赋值运算符
    str3.print(); // 输出 "Hello"
    str1.print(); // 输出 "(null)"

    return 0;
}

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1567

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

241

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

150

2025.10.17

Go语言实现运算符重载有哪些方法
Go语言实现运算符重载有哪些方法

Go语言不支持运算符重载,但可以通过一些方法来模拟运算符重载的效果。使用函数重载来模拟运算符重载,可以为不同的类型定义不同的函数,以实现类似运算符重载的效果,通过函数重载,可以为不同的类型实现不同的操作。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

196

2024.02.23

Go语言实现运算符重载有哪些方法
Go语言实现运算符重载有哪些方法

Go语言不支持运算符重载,但可以通过一些方法来模拟运算符重载的效果。使用函数重载来模拟运算符重载,可以为不同的类型定义不同的函数,以实现类似运算符重载的效果,通过函数重载,可以为不同的类型实现不同的操作。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

196

2024.02.23

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

1

2026.03.13

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

39

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

140

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

47

2026.03.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 11.2万人学习

C 教程
C 教程

共75课时 | 5.4万人学习

C++教程
C++教程

共115课时 | 21.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号