只谈C++11新特性 - 删除函数

删除函数

背景

在 C++11 之前,C++ 的类默认会生成拷贝构造函数和赋值运算符。这在某些情况下会引发问题,尤其是在我们希望明确禁止某些操作时。假设我们有一个类,它不希望被拷贝,但未明确声明拷贝构造函数和赋值运算符,这时编译器会自动生成默认实现,导致程序员可能无意间拷贝了该类对象,造成不可预期的行为或运行时错误。

示例

cpp 复制代码
#include <iostream>

class NonCopyable {
public:
    NonCopyable() {}
    ~NonCopyable() {}
};

int main() {
    NonCopyable obj1;
    NonCopyable obj2 = obj1; // 未明确禁止拷贝,编译器生成默认的拷贝构造函数
    return 0;
}

上述代码中,NonCopyable 类没有声明拷贝构造函数,导致编译器生成了默认实现。而我们希望禁止对象的拷贝,却无法直接在编译时捕获这种问题。

常见的解决方案:将函数设为 private

在 C++11 之前,一个常见的解决方法是将不希望调用的拷贝构造函数和赋值运算符声明为 private,并且不提供实现。这样,尝试拷贝或赋值时会因访问权限问题导致编译错误。

cpp 复制代码
#include <iostream>

class NonCopyable {
public:
    NonCopyable() {}
    ~NonCopyable() {}

private:
    NonCopyable(const NonCopyable&); // 声明为 private,禁止拷贝
    NonCopyable& operator=(const NonCopyable&); // 声明为 private,禁止赋值
};

int main() {
    NonCopyable obj1;
    // NonCopyable obj2 = obj1; // 编译错误:拷贝构造函数是 private 的
    // obj1 = obj2;            // 编译错误:赋值运算符是 private 的
    return 0;
}

虽然这种方式能够达到禁止拷贝和赋值的目的,但它的缺点在于代码不够直观,而且容易因未实现的函数导致链接错误。


新特性:删除函数

C++11 引入了"删除函数"(Deleted Functions)的概念,它允许开发者显式地声明某些函数为"已删除",以禁止这些函数的调用。这一特性为类的设计提供了更精确的控制。

通过将函数声明为 = delete,编译器会在编译期检查是否存在对这些函数的调用,并在调用时产生编译错误。


删除函数的用途

禁止拷贝或赋值

通过删除拷贝构造函数和赋值运算符,可以明确禁止类的拷贝和赋值。

示例代码:禁止拷贝

cpp 复制代码
#include <iostream>

class NonCopyable {
public:
    NonCopyable() {}
    ~NonCopyable() {}

    // 禁止拷贝构造
    NonCopyable(const NonCopyable&) = delete;

    // 禁止赋值运算
    NonCopyable& operator=(const NonCopyable&) = delete;
};

int main() {
    NonCopyable obj1;
    // NonCopyable obj2 = obj1; // 编译错误:调用已删除的拷贝构造函数
    // obj1 = obj2;            // 编译错误:调用已删除的赋值运算符
    return 0;
}

删除函数的优点

  1. 显式性:代码更清晰,开发者可以直接通过声明了解哪些操作被禁止。
  2. 安全性:在编译时检测不正确的调用,避免运行时错误。
  3. 可读性 :比通过私有继承 boost::noncopyable 或声明私有拷贝构造函数的旧方式更直观。

总结

C++11 的删除函数为开发者提供了一种优雅且高效的方式来明确禁止特定操作,特别是在需要防止类的拷贝或赋值时。这一特性提升了代码的安全性和可读性,使得开发者能够更好地设计和维护复杂的 C++ 项目。


相关推荐
Dicky-_-zhang几秒前
日志管理实战:ELK与Loki对比选型与落地实践
java·jvm
nJI74egg116 分钟前
JavaEE初阶---《JUC 并发编程完全指南:组件用法、原理剖析与面试应答》
java·面试·java-ee
刮风那天24 分钟前
Android AMS创建进程不用Binder而用Socket?
android·java·binder
小王C语言25 分钟前
【线程概念与控制】:线程封装
jvm·c++·算法
程序员老邢30 分钟前
【技术底稿 37】Spring Boot 3.x 自动装配 “死锁” 排查:3 个注解实现条件化装配与 Mock 兜底
java·spring boot·后端·自动装配·rag·技术底稿
学习,学习,在学习31 分钟前
Qt工控仪器程序框架设计详解(工控多仪器控制版本)
开发语言·c++·qt
三品吉他手会点灯37 分钟前
C语言学习笔记 - 35.数据类型 - printf函数的非输出控制符与格式优化
c语言·开发语言·笔记·学习
日月云棠1 小时前
JAVA数据结构与算法 - 基础:链表
java·后端
日月云棠1 小时前
JAVA数据结构与算法 - 基础:栈 (Stack) 深度解析
java·后端
xiguolangzi1 小时前
java使用Map映射遍历方法
java·后端