【读书笔记】Effective C++ 条款5~6:若不想使用编译器自动生成的函数,就该明确拒绝

目录

[一、C++ 默认生成的函数](#一、C++ 默认生成的函数)

二、何时会拒绝生成?

三、明确拒绝拷贝的方法

[1. 将拷贝函数放到 private 中(但不定义)](#1. 将拷贝函数放到 private 中(但不定义))

[2. 友元也无法绕过](#2. 友元也无法绕过)

四、通过继承实现(更优雅的方式)

[1. 定义 uncopyable 基类](#1. 定义 uncopyable 基类)

[2. 派生类继承 nocopy](#2. 派生类继承 nocopy)

[3. 原理](#3. 原理)


一、C++ 默认生成的函数

如果不显式声明,C++ 编译器会自动生成以下函数(都是 public 且 inline 的):

  • 默认构造函数

  • 拷贝构造函数

  • 拷贝赋值运算符

  • 析构函数

二、何时会拒绝生成?

在某些情况下,编译器可能拒绝生成默认的拷贝函数:

  1. 类中有 const 成员:const 成员不能被赋值,因此拷贝赋值运算符需要自定义

  2. 类中有引用成员:引用必须在初始化时绑定,不能被重新赋值

  3. 将拷贝函数放到 private 中:手动阻止生成和调用

这些情况不违背 C++ 规则,只是需要程序员自己定义合适的拷贝行为。

三、明确拒绝拷贝的方法

1. 将拷贝函数放到 private 中(但不定义)

cpp 复制代码
class hause {
public:
    hause(double s, const std::string& name) : _s(s), _name(name) {}

private:
    // 只声明不定义,阻止编译器自动生成
    hause& operator=(const hause&);
    hause(const hause&);

    double _s;
    std::string _name;
};

说明

  • 每套房子都是唯一的,因此不允许拷贝

  • 将拷贝构造函数和拷贝赋值运算符声明为 private,阻止了外部调用

  • 只声明不定义 :如果成员函数或友元试图调用,会得到链接错误

2. 友元也无法绕过

cpp 复制代码
class hause {
    friend hause func(hause& h);  // 声明友元函数

public:
    hause(double s, const std::string& name) : _s(s), _name(name) {}
    std::string getname() { return _name; }

private:
    hause& operator=(const hause&);  // 只声明不定义
    hause(const hause&);

    double _s;
    std::string _name;
};

hause func(hause& h) {
    hause ret = h;  // 调用拷贝构造函数,只有声明没有定义
    std::cout << ret.getname() << std::endl;
    return ret;
}

结果

  • 虽然 funchause 的友元,可以访问 private 成员

  • 但拷贝函数只有声明,没有定义

  • 因此链接时找不到函数定义,产生链接错误

典型例子iostream 类家族就通过这种方式阻止拷贝。比如不能拷贝 coutcin 对象。

四、通过继承实现(更优雅的方式)

1. 定义 uncopyable 基类

cpp 复制代码
class nocopy {
public:
    nocopy() {}                         // 允许构造和析构
    ~nocopy() {}

private:
    nocopy& operator=(const nocopy&);    // 只声明不定义
    nocopy(const nocopy&);
};

2. 派生类继承 nocopy

cpp 复制代码
class hause : public nocopy {
public:
    hause(double s, const std::string& name) : _s(s), _name(name) {}
    std::string getname() { return _name; }

private:
    double _s;
    std::string _name;
};

3. 原理

当尝试拷贝 hause 对象时:

  1. 编译器会尝试生成 hause 的拷贝构造函数

  2. 生成的拷贝构造函数会先调用基类 nocopy 的拷贝构造函数

  3. 但基类 nocopy 的拷贝构造函数是 private 且未定义

  4. 因此编译失败(如果是在成员函数内尝试拷贝,则产生链接错误)

优点

  • 更清晰地表达意图:hause 类"不可拷贝"

  • 不需要在每个要禁止拷贝的类中重复声明 private 拷贝函数

  • 错误信息通常在编译期(而不是链接期)就能发现

相关推荐
隔窗听雨眠8 分钟前
ORM框架选型指南:MyBatis与Hibernate的全面对比
java·开发语言·数据库
-凌凌漆-12 分钟前
【Qt】C++中protected与private的区别
开发语言·c++·qt
j7~16 分钟前
【C++】类和对象(上)--带你全面理解类和对象的概念,以及this指针的理解和相关面试题
java·开发语言·封装·this指针·类的实例化·访问限定符·类的命名
叶帆16 分钟前
【YFIOs】用C#开发硬件之串口通信
开发语言·c#
于先生吖17 分钟前
同城物流创业项目,Java源码搭建多车型搬家拉货、就近配货预约小程序
java·开发语言·小程序
码不停蹄的玄黓18 分钟前
Java 异常分类
java·开发语言
牛油果子哥q21 分钟前
【C++前置声明与头文件】C++前置声明与头文件深度精讲:重复包含、循环依赖、重复定义报错、工程编译架构与实战解决方案
开发语言·c++
-凌凌漆-22 分钟前
Qt QML应用层框架
开发语言·qt
少司府22 分钟前
C++进阶:map和set的使用
开发语言·数据结构·c++·容器·stl·set·map
江湖中的阿龙23 分钟前
23种设计模式
java·开发语言·设计模式