C++ 禁用类的拷贝构造函数和赋值运算符

C++中如果没有显式定义类的构造函数和赋值运算符,编译器会自动生成对应的函数,但是对于一些含有指针成员变量的类,自动生成的成员函数只会进行浅拷贝,会导致动态申请的内存在对象析构的时候double free,引起崩溃的问题。
因此如果没有必要,通常会禁用该接口,避免用户调用该接口造成问题。
禁用的方式主要有两种:

一、C++11 中可以直接使用 =delete 来禁用该接口

class Basic
{
public:
    Basic();

    // 禁用拷贝构造函数
    Basic(const Basic& bs) = delete;
    // 禁用赋值运算符
    Basic& operator=(const Basic& bs) = delete;

    ~Basic();

private:
    int num1;
    int* p1;
};

此时用户将无法调用该接口,否则会编译报错
delete 除了可以应用于类的成员函数,也可以应用于非成员函数


二、将其声明为私有的成员函数

另一种方式就是将其声明为私有的成员函数,然后只是声明,无需进行定义(C++中的函数可以只进行声明,而无需定义,只要该函数不被调用就可以正常编译,如果被调用的话,就会出现链接错误)。

class Basic
{
public:
    Basic();
    ~Basic();

private:
    Basic(const Basic& bs); // 拷贝构造函数声明为私有
    Basic& operator=(const Basic& bs); // 赋值运算符声明为私有

private:
    int num1;
    int* p1;
};

通过这种方式外部将无法调用该接口


参考资料:
Prefer deleted functions to private undefined ones.

posted @ 2024-08-09 09:20  Jeffxue  阅读(19)  评论(0编辑  收藏  举报