概述

这一篇简单介绍一些其他的比较实用的特性,如果读者想了解现代C++的全部特性,参考:cpp reference

其他特性

预置和弃置函数default&delete

在 C++11 中引入了 defaultdelete 关键字,允许程序员更加明确地控制类的默认操作(如默认构造函数,拷贝构造函数,拷贝赋值运算符,析构函数等)。

default

default 关键字用于明确地要求编译器生成默认的实现。例如,如果你想要一个类拥有默认的构造函数,你可以这样做:

class Obj {
public:
Obj() = default; // 使用编译器生成的默认构造函数
};

delete

delete 关键字用于禁止编译器生成默认的实现。例如,如果你不希望你的类被拷贝,你可以这样做:

class Obj {
public:
Obj(const Obj&) = delete; // 禁止拷贝构造函数
Obj& operator=(const Obj&) = delete; // 禁止拷贝赋值运算符
};

在这个例子中,如果你尝试拷贝 Obj 的实例,编译器将会报错。

这两个关键字可以让你更明确地控制类的行为,防止编译器生成你不希望的默认操作。

继承和多态的控制final&override

override

override 关键字用于明确表示一个虚函数覆盖了它的基类中的版本。这可以帮助编译器检查你的代码,防止你因为拼写错误或参数不匹配而无意中没有覆盖基类的函数。例如:

class Base {
public:
virtual void foo(int) {}
}; class Derived : public Base {
public:
void foo(int) override; // 明确表示这个函数覆盖了基类的版本
};

final

final 关键字可以用于类或虚函数。当用于类时,它表示这个类不能被继承。例如:

class Base final {};  // 这个类不能被继承

class Derived : public Base {};  // 错误:Base 是 final 的

final 用于虚函数时,它表示这个函数在派生类中不能被覆盖。例如:

class Base {
public:
virtual void foo() final; // 这个函数不能被覆盖
}; class Derived : public Base {
public:
void foo(); // 错误:Base::foo 是 final 的
};

这两个关键字可以更明确地控制类的继承和多态行为,防止错误的覆盖或继承。

有作用域的枚举

在 C++11 之前,枚举类型的值可以隐式转换为整数,而且枚举类型的成员在枚举类型的作用域之外是可见的。这可能会导致命名冲突和类型安全问题。

作用域枚举通过使用 enum class 关键字来定义,如下所示:

enum class Color {
Red,
Green,
Blue
};

作用域枚举的成员只能通过作用域解析运算符 :: 来访问,这可以避免命名冲突。例如,不能直接写 Red,而应该写 Color::Red

此外,作用域枚举的值不能隐式转换为整数,这有助于提高类型安全

列表初始化

C++11 引入了一种新的初始化语法,通常被称为列表初始化或统一初始化。这种语法使用花括号 {} 来初始化对象,可以用于几乎所有的初始化情况。

如:

int a = {5};  // 初始化基本类型
std::string s = {"hello"}; // 初始化类类型
int arr[] = {1, 2, 3, 4, 5}; // 初始化数组
std::vector<int> v = {1, 2, 3, 4, 5}; // 初始化容器
struct Point {int x, y;} p = {5, 10}; // 初始化聚合类型

列表初始化有几个优点:

  1. 统一的语法:列表初始化可以用于所有的初始化情况,包括基本类型、数组、容器、聚合类型等。

  2. 防止窄化转换:列表初始化不允许窄化转换,即从一个类型到另一个类型的转换可能丢失信息。如,试图用浮点数初始化一个整数,或者用一个大的整数初始化一个小的整数,这样的代码将无法通过编译。

int a = {3.14};  // 错误:窄化转换
char c = {300}; // 错误:窄化转换
  1. 初始化类的成员:在类的构造函数的初始化列表中,可以使用列表初始化来初始化类的成员。
class MyClass {
public:
std::vector<int> v;
MyClass() : v{1, 2, 3, 4, 5} {} // 使用列表初始化来初始化类的成员
};

列表初始化是一种非常有用的特性,可以帮助你编写更清晰、更安全的代码。

nullptr 空指针

在C++11之前,我们通常使用NULL来表示空指针。然而,NULL其实就是整数0,这可能会导致一些问题。例如,当函数重载中有一个接受int参数的版本和一个接受指针参数的版本时,如果你传递NULL给这个函数,编译器会选择接受int参数的版本,这可能不是你想要的结果。

为了解决这个问题,C++11引入了nullptr关键字来表示空指针。nullptr的类型是nullptr_t,它可以隐式转换为所有的指针类型,但不能转换为其他的类型。这使得nullptr在函数重载中的行为更符合预期。

下面是一个例子:

void f(int) {
std::cout << "f(int) called" << std::endl;
} void f(char*) {
std::cout << "f(char*) called" << std::endl;
} int main() {
f(NULL); // 输出 "f(int) called"
f(nullptr); // 输出 "f(char*) called"
return 0;
}

在这个例子中,当你传递NULLf函数时,编译器选择了接受int参数的版本。但是当你传递nullptrf函数时,编译器选择了接受指针参数的版本。这是因为nullptr的类型是nullptr_t,它可以隐式转换为char*,但不能转换为int

因此,C++11推荐使用nullptr来表示空指针,而不是NULL或者0。

noexcept 不会抛出异常承诺

noexcept是C++11引入的一个新关键字,用于指定函数是否会抛出异常。如果一个函数被声明为noexcept,那么它保证不会抛出任何异常。如果在运行时该函数确实抛出了异常,那么程序将调用std::terminate来立即结束执行。

如:

void f() noexcept {
// 这个函数保证不会抛出任何异常
} void g() {
throw std::runtime_error("An error occurred"); // 这个函数可能会抛出异常
}

使用noexcept关键字有两个主要的好处:

  1. 性能优化:如果编译器知道一个函数不会抛出异常,那么它可能会生成更优化的代码。例如,一些需要异常安全保证的操作(如对象的移动)在知道不会抛出异常的情况下,可以被编译器优化。

  2. 提供更清晰的接口:通过在函数声明中使用noexcept关键字,你可以明确地告诉调用者该函数不会抛出任何异常。这可以帮助调用者更好地理解函数的行为,并据此来编写代码。

需要注意的是,noexcept是一个承诺,如果声明一个函数为noexcept,那么你需要确保它在任何情况下都不会抛出异常。如果不能保证这一点,最好不要使用noexcept关键字。

另外,你也可以使用noexcept运算符来检查一个表达式是否保证不抛出异常:

static_assert(noexcept(f()));  // 编译时检查f()是否不抛出异常

三路比较(c++ 20)

假设我们有一个 Person 类,它有一个名字和年龄属性。我们想要比较两个 Person 对象,首先比较他们的名字,如果名字相同,再比较他们的年龄。

在C++20中,我们可以使用三路比较运算符 <=> 来实现这个比较逻辑

#include <string>
#include <compare> struct Person {
std::string name;
int age; auto operator<=>(const Person& other) const {
if (auto cmp = name <=> other.name; cmp != 0) {
return cmp;
}
return age <=> other.age;
}
}; #include <iostream> int main() {
Person alice{"Alice", 20};
Person bob{"Bob", 20};
Person charlie{"Charlie", 25}; std::cout << ((alice <=> bob) < 0 ? "Alice < Bob" : "Alice >= Bob") << std::endl;
std::cout << ((alice <=> charlie) < 0 ? "Alice < Charlie" : "Alice >= Charlie") << std::endl;
std::cout << ((bob <=> charlie) < 0 ? "Bob < Charlie" : "Bob >= Charlie") << std::endl;
}

operator<=> 首先比较 name,如果 name 不相同,就返回 name 的比较结果;如果 name 相同,就比较 age,并返回 age 的比较结果。我们也可以使用auto operator<=>(const Person& other) const = default让编译器生成,它会按照成员的声明顺序比较每个成员。

值得一提的特性

  • alignof 与 alignas 内存对齐相关
  • static_assert 静态断言
  • c++17:if和switch语句中进行初始化,如if (int i = f(); i > 10) {}switch (int i = f(); i) {}
  • 聚合初始化(花括号),c++20允许实用圆括号

C++20 一些新的、未实践但感觉有用的特性

  • 协程
  • 模块
  • 限定与概念
  • 缩略函数模板
  • DR :数组 new 可推导数组长度

现代C++(Modern C++)基本用法实践:N、其他零散的常用特性的更多相关文章

  1. Linux中sed的用法实践

    Linux中sed的用法实践 参考资料:https://www.cnblogs.com/emanlee/archive/2013/09/07/3307642.html http://www.fn139 ...

  2. Java map 详解 - 用法、遍历、排序、常用API等

    尊重原创: http://www.cnblogs.com/lzq198754/p/5780165.html 概要: java.util 中的集合类包含 Java 中某些最常用的类.最常用的集合类是 L ...

  3. Linux find命令的用法实践

    一.find命令简介 Linux下find命令在目录结构中搜索文件,并执行指定的操作.Linux下find命令提供了相当多的查找条件,功能很强大.由于find具有强大的功能,所以它的选项也很多,其中大 ...

  4. PHP命令空间namespace及use的用法实践总结

    以下皆为本人自我理解内容,如有失误之处,请多多包涵. 文章大纲: 使用namespace的目的 namespace的使用方法 使用use的目的 use的使用方法 使用namespace的目的: 团队合 ...

  5. CountDownLatch用法实践

    项目中写多线程时,需要判断所有线程是否执行完毕,所以想到了添加累加器来判断.这个累加器使用什么变量,找到了以下2种方式. 1. 类似AtomicInteger这种提供原子操作的类型AtomicInte ...

  6. java反射-- Field 用法实践

    java 反射是一种常用的技术手段, 通过加载类的字节码的方式来获取相关类的一些信息 比如成员变量, 成员方法等. Field 是什么呢? field 是一个类, 位于java.lang.reflec ...

  7. ASP.NET Web API实践系列04,通过Route等特性设置路由

    ASP.NET Web API路由,简单来说,就是把客户端请求映射到对应的Action上的过程.在"ASP.NET Web API实践系列03,路由模版, 路由惯例, 路由设置"一 ...

  8. C++11常用特性介绍——for循环新用法

    一.for循环新用法——基于范围的for循环 for(元素类型 元素对象 : 容器对象) { //遍历 } 1)遍历字符串 std::string str = "hello world&qu ...

  9. C++11常用特性介绍——nullptr关键字及用法

    一.nullptr关键字及用法 1)NULL的二义性 void func(int) {} void func(int*) {} 当函数调用func(NULL)时会执行哪个函数呢? 先看C++对NULL ...

  10. HTML meta标签的用法及head中的一些常用标签

    meta是用来在HTML文档中模拟HTTP协议的响应头报文.meta主要为分HTTP标头信息(HTTP-EQUIV)和页面描述信息(NAME)标头信息包括文档类型.字符集.语言等浏览器正确显示网页的信 ...

随机推荐

  1. 探究公众号接口漏洞:从后台登录口到旁站getshell

    探究公众号接口漏洞:从后台登录口到旁站getshell 1.入口 发现与利用公众号接口安全漏洞 某120公众号提供了一处考核平台,通过浏览器处打开该网站. 打开可以看到一处密码登录口,试了一下常用的手 ...

  2. 今天能恢复我的Django吗——恢复了!

    今天能用两小时恢复我的Django吗 实在是累了,昨天和队友改bug的时候为了能在我的电脑上实现他的程序就在datagrip中删了我django建的表.没想到啊,这一删就全是报错!! 不说了,今天看看 ...

  3. C# List转SqlServer、MySql中in字符串

    var oneList = new List<string> { "1", "2", "3" }; var oneString ...

  4. 给定一个整数数组nums和一个整数目标值target,请你在该数组中找出和为目标值target的那两个整数,并返回它们的数组下标。

    /** * 给定一个整数数组nums和一个整数目标值target,请你在该数组中找出和为目标值target的那两个整数,并返回它们的数组下标. * * 你可以假设每种输入只会对应一个答案.但是,数组中 ...

  5. Godot报错 Node not found: "SubViewport"[一问随笔]

    问题: 使用TextureRect显示SubViewport的内容,结果发生了如下报错 E 0:00:01:0007 get_node: Node not found: "SubViewpo ...

  6. C# 无需管理员权限提示,操作C盘文件

    在C盘创建.移动文件,如果当前不是管理员身份,是没办法直接操作. 如果当前程序有管理员权限,那可以直接操作. 但是,添加管理员权限启动,会弹出用户确认提示框. 在某些场景下,其实是不想让用户看到这样的 ...

  7. NOIP 2021 备战计划

    NOIP 2021 备战计划 复习知识点: 加粗表示一定去复习,?表示很可能不需要 线段树.树状数组:无论最近写多少遍都要去好好复习 Dij.SPFA:理由同上 大DP:哪个不重要? 门类:线性DP. ...

  8. I-o-C 一篇概览

    一.ioC 容器和 Bean介绍 IoC(Inversion of Control )也被称之为 DI(dependency injection),名称侧重点略有不同. 所谓控制翻转即对象通过构造函数 ...

  9. 2022-12-14:给定一个正数n, 表示从0位置到n-1位置每个位置放着1件衣服 从0位置到n-1位置不仅有衣服,每个位置还摆着1个机器人 给定两个长度为n的数组,powers和rates pow

    2022-12-14:给定一个正数n, 表示从0位置到n-1位置每个位置放着1件衣服 从0位置到n-1位置不仅有衣服,每个位置还摆着1个机器人 给定两个长度为n的数组,powers和rates pow ...

  10. SQL:DATEDIFF和DATEADD函数

    DATEDIFF和DATEADD函数.DATEDIFF函数计算两个日期之间的小时.天.周.月.年等时间间隔总数.DATEADD函数计算一个日期通过给时间间隔加减来获得一个新的日期.要了解更多的DATE ...