本篇为目前c++特性的最后一篇,我们从cmake语法的教程中插入了三篇c++语言特性的介绍,主要是为了对c++近些年来的变化有一个更加全面的了解。
std::require
是一个constexpr
函数模板,用于在编译时检查某个表达式的真假值。如果表达式为真,则该函数返回一个无意义的类型 void_t
;否则编译会失败,出现相应的错误信息优点
创建模块
// cppcon.cpp
export module cppcon;
namespace CppCon {
auto GetWelcomeHelper() { return "Welcome to CppCon 2019!"; }
export auto GetWelcome() { return GetWelcomeHelper();}
}
引用模块
// main.cpp
import cppcon;
int main(){
std::cout << CppCon::GetWelcome();
}
Range
代表一串元素, 或者一串元素中的一段,类似于begin/end
对。
好处
协程定义
是一个函数,具备如下关键字之一:
应用场景
template<typename T> concept Incrementable = requires(T x) {x++; ++x;};
使用
template<Incrementable T>
void Foo(T t);
template<typename T> requires Incrementable<T>
void Foo(T t);
template<typename T>
void Foo(T t) requires Incrementable<T>;
void Foo(Incrementable auto t);
[=, this] 需要显式捕获this变量
模板形式的 Lambda 表达式
可以在lambda
表达式中使用模板语法:
[]template<T>(T x) {/* ... */};
[]template<T>(T* p) {/* ... */};
[]template<T, int N>(T (&a)[N]) {/* ... */};
优点
C++20
之前: 获取 vector
元素类型, 需要这么写
auto func = [](auto vec){
using T = typename decltype(vec)::value_type;
}
C++20
可以:
auto func = []<typename T>(vector<T> vec){
// ...
}
lambda
形参类型, 访问静态函数
:C++20之前:
auto func = [](auto const& x){
using T = std::decay_t<decltype(x)>;
T copy = x; T::static_function();
using Iterator = typename T::iterator;
}
C++20以后:
auto func = []<typename T>(const T& x){
T copy = x; T::static_function();
using Iterator = typename T::iterator;
}
C++20之前:
auto func = [](auto&& ...args) {
return foo(std::forward<decltype(args)>(args)...);
}
C++20以后:
auto func = []<typename …T>(T&& …args){
return foo(std::forward(args)...);
}
C++20之前
template<class F, class... Args>
auto delay_invoke(F f, Args... args){
return [f, args...]{
return std::invoke(f, args...);
}
}
C++20以后
template<class F, class... Args>
auto delay_invoke(F f, Args... args){
// Pack Expansion: args = std::move(args)...
return [f = std::move(f), args = std::move(args)...](){
return std::invoke(f, args...);
}
}
constexpr
虚函数constexpr
的虚函数可以重写非 constexpr
的虚函数constexpr
虚函数可以重写 constexpr
的虚函数dynamic_cast()
和 typeid
union
成员的值try/catch
throw
语句try/catch
不发生作用constexpr std::vector
constexpr string &vector
std::string
和 std::vector
类型现在可以作为 constexpr
constexpr
反射智能指针
(shared_ptr)线程安全
问题:
将智能指针
变成线程安全
:
mutex
控制智能指针的访问std::atomic_load()
, atomic_store()
, …C++20
: atomic
, atomic
:
mutex
deprecated
)自动合流
void DoWorkPreCpp20() {
std::thread job([] { /* ... */ });
try {
// ... Do something else ...
} catch (...) {
job.join();
throw; // rethrow
}
job.join();
}
void DoWork() {
std::jthread job([] { /* ... */ });
// ... Do something else ...
} // jthread destructor automatically calls join()
中断
std::jthread job([](std::stop_token token) {
while (!token.stop_requested()) {
//...
}
});
//... job.request_stop();
// auto source = job.get_stop_source()
// auto token = job.get_stop_token()
在传统的多线程
(进程
)的编程中,处理数据共享是一个重中之重。目前流行的多核(多CPU)编程中,虽然采用了更多的分布式
的算法,但最终细分到一个处理单元中,仍然是处理线程间数据的拆分。即,通过数据结构
的设计和算法
的分拆,实现最小的数据冲突结果。
解决多线程
编程中的一个重要的问题就是如何处理数据的同步问题,如有mutex
,event
,condition
等等。也有的会提到c++11
后的lock
等。
在c++20
中增加了以下几类同步
数据结构
:
信号量(Semaphore)
轻量级的同步原语,可以实现 mutex
, latches
, barriers
, …等同步数据结构。
两种表现类型:
多元信号量
(counting semaphore): 建模非负值资源计数二元信号量
(binary semaphore): 只有两个状态的信号量主要方法有:
std::atomic 等待和通知接口
等待
/阻塞
在原子对象直到其值发生改变, 然后通知函数发送通知,它比单纯的自旋锁和轮询要效率高。
主要方法有:
这个其实是实现CAS
的,在以前就有,在c++20
中又增加了相关的一些具体的实现罢了。
锁存器(Latches)
latch
是 std::ptrdiff_t
类型的向下计数器,它能用于同步线程
。在创建时初始化计数器的值。线程可能在 latch
上阻塞直至计数器减少到零。没有可能增加或重置计数器,这使得 latch
为单次使用的屏障。同时调用 latch
的成员函数,除了析构函数
,不引入数据竞争。
注意:它区别于下面的Barriers
的是它只有使用一次。
屏障(Barriers)
std::barrier
提供允许至多为期待数量的线程阻塞直至期待数量的线程到达该屏障。不同于 std::latch
,屏障可重用:一旦到达的线程从屏障阶段的同步点除阻,则可重用同一屏障。
std::atomic_ref
类型对其引用的对象进行原子操作。
使用std::atomic_ref
进行多线程读写时不会造成数据争用。被引用对象的生命周期必须超过std::atomic_ref
。操作std::atomic_ref
的子对象是未定义行为。
指定初始化(Designated Initializers)
struct Data {
int anInt = 0;
std::string aString;
};
Data d{ .aString = "Hello" };
三路比较运算符 <=>
// 如果 a < b 则为 true
(a <=> b) < 0
// 如果 a > b 则为 true
(a <=> b) > 0
// 如果 a 与 b 相等或者等价 则为 true
(a <=> b) == 0
标准库类型支持: vector
, string
, map
, set
, sub_match
, …
范围 for 循环语句支持初始化语句
switch
语句初始化 (C++17
):
struct Foo {
int value;
int result;
};
Foo GetData() {
return Foo();
}
int main() {
switch (auto data = GetData(); data.value) {
case 1:
return data.result;
}
}
if
语句初始化 (C++17
):
struct Foo {
int value; int result;
};
Foo* GetData() {
return new Foo();
}
int main() {
if (auto data = GetData(); data) {
// Use 'data’
}
}
现在范围 for
循环同样支持初始化 (C++20
):
struct Foo {
std::vector<int> values;
};
Foo GetData() {
return Foo();
}
int main() {
for (auto data = GetData(); auto& value : data.values) {
// Use 'data’
}
}
非类型模板形参支持字符串
template<auto& s> void DoSomething() {
std::cout << s << std::endl;
}
int main() {
DoSomething<"CppCon">();
}
[[likely]], [[unlikely]]
先验概率指导编译器优化
switch (value) {
case 1: break;
[[likely]] case 2: break;
[[unlikely]] case 3: break;
}
日历(Calendar)和时区(Timezone)功能
具体操作和相关类型请参考其他示例。
std::span
定义:某段连续数据的”视图”
特性:
string_view
)stride
)// fixed-size: 42 ints
int data[42]; span<int, 42> a {data};
// dynamic-size: 42 ints
span<int> b {data};
// compilation error
span<int, 50> c {data};
// dynamic-size: len ints
span<int> d{ ptr, len };
特性测试宏
通过它可以判断编译器是否支持某个功能。
语言特性:
__has_cpp_attribute(fallthrough)
__cpp_binary_literals
__cpp_char8_t
__cpp_coroutines
标准库特性:
__cpp_lib_concepts
__cpp_lib_ranges
__cpp_lib_scoped_lock
包含 C++ 标准库
版本, 发布日期
, 版权证书
, 特性宏
等。
consteval 函数
constexpr
函数可能编译期执行, 也可以在运行期执行, consteval
只能在编译器执行, 如果不满足要求编译不通过。
constinit: 强制指定以常量方式初始化
const char* GetStringDyn() {
return "dynamic init";
}
constexpr const char* GetString(bool constInit) {
return constInit ?
"constant init" :
GetStringDyn();
}
// ✔
constinit const char* a = GetString(true);
// ❌
constinit const char* b = GetString(false);
用 using 引用 enum 类型
enum class CardTypeSuit {
Clubs,
Diamonds,
Hearts,
Spades
};
std::string_view GetString(const CardTypeSuit cardTypeSuit) {
switch (cardTypeSuit) {
case CardTypeSuit::Clubs:
return "Clubs";
case CardTypeSuit::Diamonds:
return "Diamonds";
case CardTypeSuit::Hearts:
return "Hearts";
case CardTypeSuit::Spades:
return "Spades";
}
}
std::string_view GetString(const CardTypeSuit cardTypeSuit) {
switch (cardTypeSuit) {
using enum CardTypeSuit;
case Clubs: return "Clubs";
case Diamonds: return "Diamonds";
case Hearts: return "Hearts";
case Spades: return "Spades";
}
}
格式化库(std::format)
std::string s = std::format("Hello CppCon {}!", 2019);
增加数学常量
包含 e
, log2e
, log10e
, pi
, inv_pi
, inv_sqrt
ln2
, ln10
, sqrt2
, sqrt3
, inv_sqrt3
, egamma
std::source_location
用于获取代码位置, 对于日志和错误信息尤其有用
[[nodiscard(reason)]]
表明返回值不可抛弃, 加入理由的支持
[[nodiscard("Ignoring the return value will result in memory leaks.")]]
void* GetData() { /* ... */ }
位运算
加入循环移位, 计数0
和1
位等功能
一些小更新
starts_with
, ends_with
map
支持 contains
查询是否存在某个键list
和 forward list
的 remove
, remove_if
和 unique
操作返回 size_type
表明删除个数shift_left
, shift_right
midpoint
计算中位数, 可避免溢出lerp
线性插值 lerp( float a, float b, float t )
返回unsequenced_policy
(execution::unseq
)std::string str = "Hello world!";
// starts_with, ends_with
bool b = str.starts_with("Hello");
std::map myMap{ std::pair{1, "one"s}, {2, "two"s}, {3, "three"s} };
// contains, 再也不用 .find() == .end() 了
bool result = myMap.contains(2);
结语:好不容易更完了C++特性的简要介绍,而接踵而来的c++ 23
最新的消息也纷至沓来,技术的更新让我们在使用语言特性上更加便利,但是其中每个技术点以及其底层实现更是值得深究和学习。 最后祝大家变得更强(cite@王骁Albert)。
Peace & Love!!!