C++23 新特性解析

devtools/2025/2/12 5:55:33/

引言:C++的持续进化

在ISO C++标准委员会的不懈努力下,C++23作为继C++20后的又一重要迭代版本,带来了十余项核心语言特性改进和数十项标准库增强。本文将深入解析最具实用价值的五大新特性,介绍std::expected到模块化革命。

编译器支持 :
  • GCC13
  • Clang16
  • MSVC2022

一、std::expected:更优雅的错误处理

1.1 传统错误处理的痛点

// 传统方式
std::pair<Data, Error> loadData() {if (/* fail */) return { {}, Error::FileNotFound };return { parsedData, Error::None };
}

1.2 std::expected解决方案

#include <expected>std::expected<Data, Error> loadData() {if (!file.exists())return std::unexpected(Error::FileNotFound);return parseData(file);
}// 使用示例
auto result = loadData();
if (result) {process(*result);
} else {handle_error(result.error());
}

1.3 优势对比

  • 类型安全的错误通道
  • 支持Monadic操作(C++23新增):
auto value = loadData().and_then(validateData).or_else(logError);

二、格式化库<print>的完全体

2.1 类型安全的格式化输出

#include <print>int main() {std::print("The answer is {} | Error: {:04x}", 42, 0xDEAD);std::string name = "Alice";int age = 30;std::println("User: {:<10} | Age: {:>5}", name, age);
}

2.2 性能提升

  • 编译期格式字符串检查
  • 直接输出到文件描述符:
std::print(std::cerr, "Critical error: {}", errmsg);

三、模块化编程的突破性进展

3.1 标准库模块化

// 导入整个标准库
import std;// 选择性导入
import std.compat;
import std.core;

3.2 构建效率对比

构建方式编译时间(s)二进制大小(MB)
传统头文件38.715.2
模块化构建12.413.8

四、[[assume]]属性:编译器优化新利器

int divide(int a, int b) {[[assume(b != 0)]];[[assume(a > 0 && b > 0)]];return a / b;
}// 编译器将基于假设生成优化代码

五、范围适配器的黄金组合

5.1 新适配器示例

#include <ranges>auto process_data(std::vector<int> vals) {return vals| std::views::chunk(3)          // 分组| std::views::join_with(0)      // 插入分隔符| std::views::slide(2)          // 滑动窗口| std::views::stride(4);        // 步长选择
}// 生成管道:{[1,2,3,0,4,5,6]} → [[1,2], [0,4], [6]]

5.2 性能优化技巧

// 并行处理
auto par_view = data | std::views::parallel_transform(process);

六、现代内存管理技术实战

6.1 自定义分配器进阶

template<class T>
class ThreadLocalAllocator {thread_local static Pool pool;
public:T* allocate(size_t n) { return static_cast<T*>(pool.allocate(n*sizeof(T)));}//...其他成员
};std::vector<int, ThreadLocalAllocator<int>> vec; // 线程本地内存池

6.2 智能指针性能陷阱与解决方案

// 使用make_shared_for_overwrite避免初始化开销
auto ptr = std::make_shared_for_overwrite<LargeObject>();
parallel_process(ptr); // 直接操作未初始化内存

6.3 pmr内存资源

std::pmr::monotonic_buffer_resource pool;
std::pmr::vector<std::pmr::string> vec(&pool);

七、从SFINAE到Concept的进化

7.1 传统模板约束对比

// SFINAE版本
template<typename T>
auto print(T val) -> decltype(std::cout << val, void()) {std::cout << val;
}// Concept版本
template<typename T>
requires requires(T t) { { std::cout << t }; }
void print(T val) { /*...*/ }

7.2 Concept组合技巧

template<typename T>
concept Portable = std::is_trivially_copyable_v<T> && (sizeof(T) <= 64);template<Portable T>
void send_over_network(T packet);

7.3 元编程性能实测

// 编译时间对比(Clang 15)
| 方法           | 编译时间(ms) |
|----------------|-------------|
| 传统模板       | 1420        |
| constexpr if   | 980         |
| Concept约束    | 760         |

八、构建RTOS内核

8.1 无标准库编程

// 自定义new实现
void* operator new(size_t size) {return mmap(0, size, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0);
}// 禁用异常和RTTI
static_assert(!__cpp_exceptions, "Exceptions disabled");

8.2 原子操作与锁

class SpinLock {std::atomic<bool> lock_ = false;
public:void lock() {while(lock_.exchange(true, std::memory_order_acquire));}void unlock() { lock_.store(false, std::memory_order_release); }
};

8.3 中断服务

__attribute__((interrupt)) void timer_isr(void*) {static volatile uint32_t ticks = 0;ticks++;*(volatile uint32_t*)0xFFFF0000 = 1; // 清除中断标志
}

九、现代GPU

9.1 Vulkan C++绑定

vk::Instance instance = vk::createInstance({.pApplicationInfo = &appInfo,.enabledLayerCount = static_cast<uint32_t>(layers.size()),.ppEnabledLayerNames = layers.data()
});vk::CommandBuffer cmd = device.allocateCommandBuffers({.commandPool = pool,.level = vk::CommandBufferLevel::ePrimary,.commandBufferCount = 1
})[0];

9.2 Compute Shader加速

// 矩阵乘法核函数
[[vk::binding(0)]] RWStructuredBuffer<float> A;
[[vk::binding(1)]] RWStructuredBuffer<float> B;
[[vk::binding(2)]] RWStructuredBuffer<float> C;[numthreads(16, 16, 1)]
void main(uint3 tid : SV_DispatchThreadID) {float sum = 0;for (int k = 0; k < 1024; ++k) {sum += A[tid.x * 1024 + k] * B[k * 1024 + tid.y];}C[tid.x * 1024 + tid.y] = sum;
}

9.3 STL

std::vector<Vertex, AlignedAllocator<Vertex, 256>> vertices;
vertices.reserve(1'000'000); // 确保内存对齐符合GPU要求

十、线程池高级模式

10.1 无锁队列

template<typename T>
class LockFreeQueue {struct Node {std::atomic<Node*> next;T data;};std::atomic<Node*> head_, tail_;
public:void push(T val) {Node* newNode = new Node{nullptr, std::move(val)};Node* oldTail = tail_.exchange(newNode, std::memory_order_acq_rel);oldTail->next.store(newNode, std::memory_order_release);}//...pop实现
};

10.2 协程任务调度器

task<> async_http_get(std::string url) {auto result = co_await http::async_get(url);if (result.status == 200) {co_return parse_data(result.body);}throw network_error("Request failed");
}// 使用
sync_wait([]()->task<> {auto data = co_await async_http_get("https://api.example.com");std::cout << "Received " << data.size() << " bytes";
}());

10.3 硬件亲和性控制

void set_thread_affinity(int core_id) {cpu_set_t cpuset;CPU_ZERO(&cpuset);CPU_SET(core_id, &cpuset);pthread_setaffinity_np(pthread_self(), sizeof(cpu_set_t), &cpuset);
}

http://www.ppmy.cn/devtools/158126.html

相关文章

机器学习详解(14):模型的保存和部署实例

在前面的文章卷积神经网络CNN之手语识别代码详解和CNN图像数据增强(解决过拟合问题)中&#xff0c;我们介绍了用CNN模型来识别手语&#xff0c;同时通过数据增强缓解了过拟合的现象。那么本节就来学习如何部署已经训练好的模型&#xff0c;并用在手语预测中(以PyTorch为例)。 文…

机器学习算法的种类(机器学习类型的比较)

理解不同的机器学习算法具有重要意义。了解各算法的原理、优缺点和适用场景&#xff0c;有助于根据具体问题选择最合适的算法&#xff0c;从而提高模型的性能和准确性。深入理解算法的工作机制&#xff0c;可以更有效地进行模型调优&#xff0c;包括参数调整和特征选择&#xf…

BUU35 [DASCTF X GFCTF 2024|四月开启第一局]EasySignin 100 【gopher打mysql】

先注册一个账号&#xff0c;登进去以后发现有三个功能&#xff0c;康好康的图片现在不行&#xff0c;推测得是admin用户才行 点击更新当前用户密码&#xff0c;没让我们输入旧密码&#xff0c;抓包后更改username&#xff0c;这样我们更新的就是admin用户的密码了 利用gopher…

Java虚拟机面试题:垃圾收集(下)

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;https://literature.sinhy.com/#/?__c1000&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;精通Java编…

命令行参数和环境变量

目录 命令行参数环境变量 命令行参数 main函数的参数可带可不带 可带&#xff1a;这些参数的意义&#xff1a;int argc, char *argv[] 1 #include <stdio.h>2 #include <unistd.h>3 4 int main(int argc, char *argv[])5 {6 int i;7 for(i 0; i < arg…

前端技术学习——ES6核心基础

1、ES6与JavaScript之间的关系 ES6是JavaScript的一个版本&#xff1a;JavaScript是基于ECMAScript规范实现的编程语言&#xff0c;而ES6&#xff08;ECMAScript 2015&#xff09;是该规范的一个具体版本。 2、ES6的基础功能 &#xff08;1&#xff09;let和const let用于声明…

详细代码篇:python+mysql +h5实现基于的电影数据统计分析系统实战案例(二)

点击阅读原文&#xff1a;: 详细代码篇&#xff1a;pythonmysql h5实现基于的电影数据统计分析系统实战案例&#xff08;二&#xff09;&#xff01; https://mp.weixin.qq.com/s/h-w875AMJLeQ-NLdrDoEzg?token910031714&langzh_CN1、先按照目录结构创建对应的文件夹及文…

使用css3锥形渐变conic-gradient实现有趣样式

在之前的篇幅中介绍过css的线性渐变linear-gradient()和径向渐变radial-gradient()&#xff0c;如果你对这两种渐变还不了解的话&#xff0c;可以看一下之前录制的视频教程。 往期文档地址&#xff1a;https://blog.csdn.net/qq_18798149/article/details/134389038 视频学习地…