C++20新特性的补充讲解

C++20 标志着 C++ 语言的一次重要更新,除了 Concepts、Ranges、协程等被广泛讨论的特性外,还有许多值得注意的改进。本文将详细探讨其他一些核心新特性,包括 constexpr 扩展、新增的 std::formatstd::spanstd::bit 操作、原子智能指针、char8_t 类型以及 lambda 表达式的改进等。通过这些特性,C++20 带来了更简洁、更高效的编程体验。

文章目录

    • @[toc]
    • 一、`constexpr` 扩展
      • 示例
    • 二、`std::format`:强大的字符串格式化
      • 示例
    • 三、`std::span`:安全的连续内存视图
      • 示例
    • 四、`std::bit` 操作库
      • 示例
    • 五、原子智能指针
      • 示例
    • 六、`char8_t` 类型
      • 示例
    • 七、lambda 表达式的改进
      • 1. 捕获 `this` 的改进
      • 2. 模板 lambda
    • 八、扩展的`[[nodiscard]]`属性
      • 示例
    • 九、改进的`std::chrono`库
      • 示例
    • 十、其他改进
      • 1. `std::source_location`
      • 2. `std::stop_token` 和 `std::jthread`
      • 3. 匿名结构体和联合体
    • 总结

一、constexpr 扩展

在 C++20 中,constexpr 关键字得到了进一步扩展,使得更多的操作可以在编译期执行。这意味着我们可以在 constexpr 中使用更复杂的代码,包括 try-catch 异常处理、动态分配和更多的标准库函数,使编译时常量计算的范围大大增加。

示例

#include <array>
#include <iostream>constexpr int factorial(int n) {if (n <= 1) return 1;return n * factorial(n - 1);
}int main() {constexpr int result = factorial(5); // 在编译期计算std::cout << result << std::endl;    // 输出:120return 0;
}

在这个例子中,我们利用 constexpr 计算阶乘函数 factorial。在 C++20 中,这种递归计算不仅更加灵活,还可以在编译期完成,为程序提供了额外的性能优化空间。


二、std::format:强大的字符串格式化

std::format 是 C++20 引入的一个重要新特性,类似于 Python 的 f-string 或者 printf 函数,提供了更现代、灵活的字符串格式化功能。std::format 使用占位符和格式说明符来简化字符串拼接操作,使代码更清晰。

示例

#include <format>
#include <iostream>int main() {int year = 2023;double pi = 3.14159;std::string formatted = std::format("The year is {} and Pi is {:.2f}.", year, pi);std::cout << formatted << std::endl;  // 输出:The year is 2023 and Pi is 3.14.return 0;
}

这里,std::format 使用 {} 作为占位符,支持指定宽度、精度等格式化参数。相比传统的 sprintf 函数,std::format 更加类型安全、功能更强大,提升了代码的可读性和安全性。


三、std::span:安全的连续内存视图

std::span 是一种轻量级的非拥有数据的容器视图,用于表示连续的内存块,例如数组或 std::vector 的某部分。std::span 提供了访问连续内存的方式,而不需要传递指针和大小参数,提高了接口的安全性和可读性。

示例

#include <span>
#include <iostream>void print(std::span<int> data) {for (int value : data) {std::cout << value << " ";}
}int main() {int arr[] = {1, 2, 3, 4, 5};print(arr); // 可以直接传入数组return 0;
}

std::span 的优势在于可以避免数组指针传递带来的边界不清问题,提供了更加一致和简洁的接口。它还可以与 std::vector 等容器结合使用,进一步增强代码的灵活性。


四、std::bit 操作库

C++20 新增了 std::bit 标准库,提供了一系列用于位操作的函数,包括按位旋转、按位计数等操作。这些操作对底层位操作有很大的帮助,简化了位处理的复杂度。

示例

#include <bit>
#include <iostream>int main() {unsigned int x = 0b0011'1100;// 位旋转操作unsigned int rotated = std::rotl(x, 2); // 左旋 2 位std::cout << "After left rotation: " << std::bitset<8>(rotated) << std::endl;// 计算 1 的数量int count = std::popcount(x);std::cout << "Number of 1s: " << count << std::endl;return 0;
}

这些位操作函数对需要处理位运算的程序员来说非常实用,提供了更高效、更简洁的位操作支持,适合于处理嵌入式系统和性能敏感的应用场景。


五、原子智能指针

C++20 引入了 std::atomic_shared_ptrstd::atomic_weak_ptr,使得智能指针能够以原子的方式操作。这一改进简化了并发编程中智能指针的管理,保证了多线程环境下的指针操作安全。

示例

#include <atomic>
#include <iostream>
#include <memory>
#include <thread>std::atomic<std::shared_ptr<int>> atomic_ptr;void thread1() {auto ptr = std::make_shared<int>(42);atomic_ptr.store(ptr);
}void thread2() {auto ptr = atomic_ptr.load();if (ptr) {std::cout << "Value: " << *ptr << std::endl;}
}int main() {std::thread t1(thread1);std::thread t2(thread2);t1.join();t2.join();return 0;
}

原子智能指针的引入简化了多线程环境下共享资源的管理,确保了数据的线程安全,是并发编程中的一项重要改进。


六、char8_t 类型

C++20 引入了 char8_t 类型,用于表示 UTF-8 编码的字符。这一类型使得代码在处理多字节字符时更加清晰和安全,避免了传统 char 类型在编码上的模糊性。

示例

#include <iostream>void print_utf8(const char8_t* text) {while (*text) {std::cout << static_cast<char>(*text);text++;}
}int main() {const char8_t* text = u8"Hello, UTF-8!";print_utf8(text); // 输出:Hello, UTF-8!return 0;
}

char8_t 提供了更强的编码明确性,减少了因编码转换产生的错误和歧义,是现代应用程序处理 UTF-8 编码的理想选择。


七、lambda 表达式的改进

C++20 为 lambda 表达式带来了几项重要改进,包括捕获 this 指针的简化和 template lambda 支持,使得 lambda 更加灵活易用。

1. 捕获 this 的改进

在 C++20 之前,lambda 捕获成员变量的方式较为繁琐。C++20 允许直接捕获 *this,避免了冗长的捕获列表。

#include <iostream>struct MyClass {int value = 42;void show() {auto lambda = [*this] { std::cout << value << std::endl; };lambda();}
};int main() {MyClass obj;obj.show(); // 输出:42return 0;
}

2. 模板 lambda

模板 lambda 允许我们定义带模板参数的 lambda 表达式,适合泛型编程场景,使得 lambda 表达式更加灵活。

#include <iostream>int main() {auto add = []<typename T>(T a, T b) {return a + b;};std::cout << add(1, 2) << std::endl;       // 输出:3std::cout << add(1.5, 2.5) << std::endl;   // 输出:4.0return 0;
}

模板 lambda 为泛型编程提供了更简洁的写法,减少了冗长的函数模板定义,让代码更具表现力。


八、扩展的[[nodiscard]]属性

C++20 扩展了 [[nodiscard]] 属性,允许对整个类型或函数标记 [[nodiscard]],表示返回值不可忽略。通过这一属性,可以避免意外忽略重要返回值引起的潜在错误。

示例

#include <iostream>[[nodiscard]] int compute_important_value() {return 42;
}int main() {compute_important_value(); // 警告:忽略了 `nodiscard` 返回值return 0;
}

[[nodiscard]] 属性提高了代码的健壮性,使得重要返回值不会被意外忽略。它适用于那些返回值不可忽略的函数,比如错误处理、资源分配等场景。通过 [[nodiscard]] 属性,C++20 能够进一步减少因忽略关键返回值导致的潜在问题。


九、改进的std::chrono

C++20 对 std::chrono 时间库进行了改进,添加了对历法和时区的支持。这些改进使得处理日期和时间更加方便,尤其是在处理跨时区和复杂日期计算的应用中。

示例

#include <iostream>
#include <chrono>
#include <format>int main() {using namespace std::chrono;// 获取当前的系统时间auto now = system_clock::now();// 格式化输出日期和时间std::cout << std::format("Current time: {:%Y-%m-%d %H:%M:%S}\n", now);// 计算未来的日期auto future_date = now + days(100);std::cout << std::format("100 days later: {:%Y-%m-%d}\n", future_date);return 0;
}

std::chrono 的改进使得日期和时间的处理更加直观,避免了开发者使用外部库来实现类似功能。通过新的时区支持,C++20 让时间计算和显示更加精确和方便。


十、其他改进

除了上面提到的主要特性,C++20 还包含了许多小而实用的改进:

1. std::source_location

std::source_location 允许获取源代码的位置(如文件名、行号和函数名),用于日志和调试信息的记录,非常适合用于调试和错误跟踪。

#include <iostream>
#include <source_location>void log(const std::string& message, const std::source_location& location = std::source_location::current()) {std::cout << "Log: " << message<< " (" << location.file_name() << ":" << location.line() << " in " << location.function_name() << ")\n";
}int main() {log("An example message");return 0;
}

2. std::stop_tokenstd::jthread

std::stop_tokenstd::jthread 简化了线程的停止机制,支持安全的线程中断。std::jthread 自动管理线程的生命周期,避免了线程资源泄漏。

#include <iostream>
#include <thread>
#include <chrono>
#include <stop_token>void task(std::stop_token stop_token) {while (!stop_token.stop_requested()) {std::cout << "Working...\n";std::this_thread::sleep_for(std::chrono::seconds(1));}std::cout << "Task stopped.\n";
}int main() {std::jthread worker(task);  // 使用 jthread 运行线程std::this_thread::sleep_for(std::chrono::seconds(3));worker.request_stop();  // 请求线程停止return 0;
}

3. 匿名结构体和联合体

C++20 支持匿名结构体和联合体,允许在类中嵌套匿名的 structunion,从而简化代码结构。

#include <iostream>struct Point {union {struct {int x, y;};int coords[2];};
};int main() {Point p;p.x = 10;p.y = 20;std::cout << "Point x: " << p.coords[0] << ", y: " << p.coords[1] << std::endl;return 0;
}

匿名结构体和联合体在需要存储多个互斥数据时提供了更便捷的方式,使代码更加紧凑。


总结

C++20 是一次全面的更新,带来了 Concepts、Ranges、Coroutines 等重量级特性,同时通过 constexpr 扩展、std::formatstd::span、原子智能指针、std::chrono 改进等大量优化,为开发者提供了更强大的工具链。这些特性不仅提升了 C++ 代码的安全性、性能和可读性,还使得开发体验更加便捷和高效。

通过这些新特性,C++20 不仅解决了长期以来的模板编程复杂性、多线程管理等问题,还引领了现代 C++ 向更加简洁、高效的方向发展。对于 C++ 开发者来说,学习和掌握这些新特性将大大提升编程效率,并为创建更具扩展性和可维护性的代码奠定基础。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/60530.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2024 年将 Postman 文档导出为 HTML 或 Markdown

2024 年将 Postman 文档导出为 HTML 或 Markdown

访问网页的全过程(知识串联)

开发岗中总是会考很多计算机网络的知识点&#xff0c;但如果让面试官只靠一道题&#xff0c;便涵盖最多的计网知识点&#xff0c;那可能就是 网页浏览的全过程 了。本篇文章将带大家从头到尾过一遍这道被考烂的面试题&#xff0c;必会&#xff01;&#xff01;&#xff01; 总…

服务器硬件介绍

计算机介绍 现在的人们几乎无时无刻都在使用电脑&#xff01;而且已经离不开电脑了。像桌上的台式电脑(桌机)、笔记本电脑(笔电)、平板电脑、智能手机等等&#xff0c;这些东西都算是电脑。 台式机电脑介绍 计算机又被称为电脑。台式机电脑主要分为主机和显示器两个部分&…

04-HTTP协议、请求报文、响应报文

欢迎来到“雪碧聊技术”CSDN博客&#xff01; 在这里&#xff0c;您将踏入一个专注于Java开发技术的知识殿堂。无论您是Java编程的初学者&#xff0c;还是具有一定经验的开发者&#xff0c;相信我的博客都能为您提供宝贵的学习资源和实用技巧。作为您的技术向导&#xff0c;我将…

基于Multisim人数出入加减计数统计电路(含仿真和报告)

【全套资料.zip】人数出入加减计数统计电路Multisim仿真设计数字电子技术 文章目录 功能一、Multisim仿真源文件二、原理文档报告资料下载【Multisim仿真报告讲解视频.zip】 功能 设计两路光控电路&#xff0c;一路放置在入口&#xff0c;另一路设置在出口&#xff0c;当有人…

车机安装第三方软件实现打开软件全屏教程

简介 越来越多的车友实现安装第三方软件了&#xff0c;但是有的车机的状态栏或者导航栏会遮挡安装的第三方软件。这样的话&#xff0c;第三方软件就会显示不全&#xff0c;体验感非常不好。所以&#xff0c;下面我教一下大家如何使用东君应用管家来实现打开第三方软件全屏。 全…

CLion配置QT开发环境

一、将qmake工程转为cmake工程&#xff08;方法一&#xff1a;用工具转换并做适当修改&#xff09; 1、工具链接&#xff1a;链接&#xff1a;https://pan.baidu.com/s/1grW2QY3sW8X2JaHWM_ePPw 提取码&#xff1a;7at4 工具源码:https://github.com/milahu/qmake2cmake 2、执行…

第3章-需求 3.4 需求的合法合规性审查

3.4 需求的合法合规性审查 3.4.1 项目需求的合法性审查3.4.2 委托研发项目的法律问题3.4.3 项目实施过程中的知识产权问题1.开发成果的知识产权保护2.开发过程中的侵权风险防控 近年来随着依法治国的深度开展&#xff0c;企业合规的概念越来越多地出现在大众眼前&#xff0c;也…

怎么保护源代码,源代码防泄密的十种方法

在现代软件开发中&#xff0c;源代码被视为企业的“数字资产”&#xff0c;它不仅承载着企业的核心竞争力&#xff0c;更是推动技术创新和业务发展的基石。然而&#xff0c;随着信息技术的普及和开发环境的复杂化&#xff0c;源代码泄密的风险也在不断增加。保护源代码的安全已…

单相锁相环,原理与Matlab实现

单相锁相环基本原理 单相锁相环的基本原理图如下所示&#xff0c; u α u_\alpha uα​ u β u_\beta uβ​经Park变换、PI控制实现对角频率 ω \omega ω和角度 θ \theta θ的估算。不同锁相环方案之间的差异&#xff0c;主要表现在正交电压 u β u_\beta uβ​的生成&#x…

腾讯云双十一福利篇之CVM云服务器

&#x1f34b;引言 随着企业数字化转型的不断深入&#xff0c;对云计算的需求日益增长。腾讯云推出的云服务器&#xff08;CVM&#xff09;为企业提供了一种灵活、高效、经济的计算服务解决方案。CVM云服务器不仅能够快速响应业务需求&#xff0c;还可以帮助企业降低前期硬件投…

DDei在线设计器-国际化

国际化 从1.2.42版开始,DDei提供了国际化支持。默认提供了中文/英文两种语言。 编辑器会读取浏览器的语言设置&#xff0c;也能够在初始化时强行指定&#xff0c;默认使用zh_CN(中文)。 本示例基于教程快速指南开发&#xff0c;如果您不清楚怎样引入DDei设计器&#xff0c;请查…

NUXT3学习日记一(在我git中拉取代码、文件讲解)

Nuxt 3 是一个基于 Vue 3 的现代框架&#xff0c;用于构建服务器端渲染&#xff08;SSR&#xff09;和静态生成的应用程序。它提供了一种简化的方式来创建高性能的 Vue 应用&#xff0c;具有许多强大的功能和优点。以下是 Nuxt 3 的一些主要应用和优点&#xff1a; 一、应用场…

小米路由器3(R3)安装mixbox命令

sh -c /extdisks/sda1/ftp/install.sh && source /etc/profile &> /dev/null install.sh文件 或者在目录新建一个粘贴 #!/bin/sh -e #copyright by monlorclear logsh() {# 输出信息到/tmp/messages和标准输出logger -s -p 1 -t "$1" "$2&quo…

DAY27|贪心算法Part01|LeetCode:455.分发饼干、376. 摆动序列、53. 最大子序和

贪心算法 贪心的本质是选择每一阶段的局部最优&#xff0c;从而达到全局最优。 贪心算法并没有固定的套路&#xff0c;最难想的就在于如何通过局部最优去推出全局最优。在做一个题目的时候&#xff0c;靠自己手动模拟&#xff0c;如果模拟可行&#xff0c;就可以试一试贪心策略…

“反卷”风暴席卷双十一,商家们却卖爆了

文丨郭梦仪 11月12日零点&#xff0c;第16届双11落下帷幕&#xff0c;但商家的故事依然在继续。 “今年双11&#xff0c;没想到能月入千万&#xff0c;流量上升了200%。”乐尘家居主理人子旭向商业数据派表示&#xff0c;他今年最明显的感受是&#xff0c;认真经营的人在这个…

OCCT7.7.0 使用VTK9.2.0

1.前言 OCCT使用的VTK版本是VTK6.1.0&#xff0c;这个版本已经比较老了&#xff0c;如果要使用新版本的VTK&#xff0c;则需要自己引入新版本VTK&#xff0c;自己编译&#xff0c;我这里也比较好奇&#xff0c;为什么OCCT不升级一下VTK版本&#xff0c;下面自己来引入VTK9.2.0…

STM32项目实战:基于STM32U5的火灾报警系统(LVGL),附项目教程/源码

《火灾报警系统_STM32U5》项目完整文档、项目源码&#xff0c;点击下方链接免费领取。 项目资料领取https://s.c1ns.cn/F5XyU STM32项目实战之“火灾报警系统”&#xff08;基于STM32U5&#xff09; 今天小编来分享一个《火灾报警系统》的项目案例&#xff0c;硬件平台是STM…

C++初阶 --- 类和对象(1)

类和对象&#xff08;1&#xff09; 一、类的相关内容1.访问限定符2.class关键字与struct关键字的区别3.类域 二、this指针三、成员函数存放在哪里&#xff1f;1. 函数的存放位置2. 练习题 一、类的相关内容 1.访问限定符 &#xff08;1&#xff09;C⼀种实现封装的⽅式&…

Python 获取PDF的各种页面信息(页数、页面尺寸、旋转角度、页面方向等)

目录 安装所需库 Python获取PDF页数 Python获取PDF页面尺寸 Python获取PDF页面旋转角度 Python获取PDF页面方向 Python获取PDF页面标签 Python获取PDF页面边框信息 了解PDF页面信息对于有效处理、编辑和管理PDF文件至关重要。PDF文件通常包含多个页面&#xff0c;每个页…