C++ 类的六个特殊成员函数

简 述: 分析 C++ Class 的六个特殊成员函数 wiki ,并且手动实现一番;

  1. 默认构造函数
  2. 析构函数
  3. 复制构造函数
  4. 复制赋值运算符
  5. 移动构造函数
  6. 移动赋值运算符

文章目录



本文初发于 “偕臧的小站”,同步转载于此。


原理

​ 特殊成员函数是类(或结构)成员函数,在某些情况下,编译器会自动为你生成。

  1. default constructor: 通常没有参数,但可以具有带默认值的参数。
  2. destructor: 销毁对象的前一刻执行清理
  3. copy constructor:新建的的对象进行初始化,形参是 const T&
  4. copy assignment constructor:已有的对象进行赋值,形参是 const T&
  5. move constructor:新建的的对象进行初始化,形参是 T &&
  6. move assignment constructor:已有的对象进行赋值,形参是 T &&

  • move 的函数相对于 copy 的函数没有 const

  • copy 是属完整的再复制拷贝一份;move 是对将亡的右值进行指针互换、节省空间提升效率

  • ”复制 / 移动构造函数“ 属于新建对象,无需判断两者是否相等;”复制 / 移动赋值运算符“ 属于已有的对象进行赋值,赋值时需要先判断两者是否相等。

  • 若有深拷贝时,”复制 / 移动赋值运算符“ 除了判断相等、数值是否有效;在拷贝字节前,还要先释放旧资源

  • 只有当类存储了需要释放的系统资源的句柄,或拥有其指向的内存的指针时,你才需要定义自定义 ”析构函数“

  • 三五法则 cppreference.com


代码

💻 win10 22H2 📎 Visual Studio 2019 📎 C++17

/*******************************************************************
 * Copyright (c) 2022~2023 XMuli  All rights reserved.
 * Description: C++ 类的六个特殊成员函数
 ******************************************************************/
#pragma once
#include <iostream>
#include <utility>
using namespace std;

class A
{
public:
    A() : m_ptr(nullptr) {
        std::cout << "default constructor" << endl;
    }

    A(const char* s) : m_ptr(nullptr) {
        std::cout << "no-default-val constructor" << endl;
        if (s) {
            auto n = std::strlen(s) + 1;
            m_ptr = new char[n];
            std::memcpy(m_ptr, s, n);
        }
    }

    A(const A& other)
        : m_ptr(nullptr) {
        std::cout << "copy constructor" << endl;
        if (other.m_ptr) {
            auto n = std::strlen(other.m_ptr) + 1;
            m_ptr = new char[n];
            std::memcpy(m_ptr, other.m_ptr, n);
        }
    }

    A& operator = (const A& other) {
        std::cout << "copy assignment constructor" << endl;
        if (this != &other) {
            delete[] m_ptr;  // Free the existing resource.  重点 !!!

            if (other.m_ptr != nullptr) {
                auto n = std::strlen(other.m_ptr) + 1;
                m_ptr = new char[n];
                std::memcpy(m_ptr, other.m_ptr, n);
            }
        }

        return *this;
    }

    A(A&& other) noexcept    // 行参无 const
        : m_ptr(nullptr) {
        std::cout << "move constructor" << endl;
        if (other.m_ptr)
            m_ptr = std::move(other.m_ptr);

        other.m_ptr = nullptr;
    }

    A& operator=(A&& other) noexcept {  // 行参无 const
        std::cout << "move assignment constructor" << endl;
        if (this != &other) {
            delete[] m_ptr;  // Free the existing resource.

            if (other.m_ptr)
                m_ptr = std::move(other.m_ptr);

            other.m_ptr = nullptr;
        }

        return *this;
    }

    ~A() {
        std::cout << "destructor" << endl;
        if (m_ptr)
            delete[] m_ptr;
    }

private:
    char* m_ptr;
};

A fn() {
    A t("A fun()");
    return t;
}

int main()
{
    A a1("a1");                // default constructor
    A a2(a1);                  // copy constructor
    A a3 = a1;                 // copy constructor
    a1 = a3;                   // copy assignment constructor

    std::cout << "----------------------------\n\n";

    //fn();                    // function returning a A object
    A a5 = std::move(a1);      // move constructor
    A a6;                      // default constructor
    a6 = std::move(a1);        // move assignment constructor
    std::cout << "Hello World!\n";

    return 0;
}

/*************************** 运行结果 *******************************
* no-default-val constructor
copy constructor
copy constructor
copy assignment constructor
----------------------------

move constructor
default constructor
move assignment constructor
Hello World!
destructor
destructor
destructor
destructor
destructor
 ******************************************************************/

拷贝构造函数是 A a2(a1); 形式,还可以是 A a3 = a1; 这种形式。关键看是新创建对象还是已有的对象赋值。move 的函数赋值,借助 std::move() 将左值转换为右值。


系列

QtExamplesSpecialMembers.h

欢迎 star ⭐ 和 fork 🍴这个系列的 C++ / QT / DTK 学习,附学习由浅入深的目录。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
C++中的静态成员函数是属于整个特殊成员函数。它可以通过名直接访问,也可以通过对象名访问。静态成员函数在其函数体内只能直接访问静态成员变量和静态成员函数,而不能访问普通数据成员和普通成员函数。静态成员函数没有隐藏的this指针。\[2\] 静态成员函数可以直接引用本中的静态成员变量,因为静态成员变量同样属于的一部分。静态成员函数主要用于访问静态成员变量,而不访问非静态成员。\[3\] 在给出的代码示例中,A中的静态成员函数Get_m1()和Get_m2()都是静态成员函数,它们可以直接访问静态数据成员m。Get_m1()通过名直接访问,而Get_m2()通过对象名访问。\[1\] 总结来说,C++中的静态成员函数是属于整个特殊成员函数,可以通过名直接访问,也可以通过对象名访问。它们在函数体内只能直接访问静态成员变量和静态成员函数,不能访问普通数据成员和普通成员函数。静态成员函数没有隐藏的this指针。静态成员函数主要用于访问静态成员变量。\[2\]\[3\] #### 引用[.reference_title] - *1* [C++——静态成员函数](https://blog.csdn.net/Getugly/article/details/84000806)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [C++中的静态成员函数](https://blog.csdn.net/SlowIsFastLemon/article/details/103664015)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [C++ 静态成员函数](https://blog.csdn.net/feng19870412/article/details/124902096)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

偕臧x

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值