0

0

如何利用c++17的并行算法(parallel algorithms)加速数据处理? (std::execution策略)

穿越時空

穿越時空

发布时间:2026-01-19 12:49:02

|

369人浏览过

|

来源于php中文网

原创

c++kquote>std::execution::par仅保证线程级并行,而par_unseq还允许编译器向量化;前者适用于顺序敏感操作,后者要求无副作用且可重排,误用会导致数据竞争或未定义行为。

如何利用c++17的并行算法(parallel algorithms)加速数据处理? (std::execution策略)

std::execution::par 和 std::execution::par_unseq 的实际区别在哪?

两者都启用多线程并行,但 std::execution::par_unseq 还允许编译器对同一迭代内操作做向量化(如 SIMD),而 std::execution::par 仅保证线程级并行,不承诺向量化。这意味着:若你的算法逻辑可安全重排(比如累加、求最大值、transform),std::execution::par_unseq 通常更快;但若涉及顺序敏感的副作用(如修改共享计数器、依赖前序迭代结果),必须用 std::execution::par,否则行为未定义。

常见误用:对含 std::cout 或 push_back() 到同一容器的操作盲目加 par_unseq —— 输出乱序、数据竞争风险极高。

哪些标准算法真正支持并行执行策略?

不是所有 函数都支持 std::execution 策略。C++17 明确要求支持并行化的有:

  • std::sortstd::stable_sort
  • std::for_eachstd::for_each_n
  • std::reducestd::transform_reduce
  • std::exclusive_scanstd::inclusive_scan
  • std::transformstd::copystd::fill
  • std::findstd::find_ifstd::search

注意:std::uniquestd::partitionstd::nth_element 等虽在标准中“鼓励实现支持”,但实际是否并行取决于 STL 实现(如 libstdc++ 12+ 和 libc++ 14+ 才逐步补全)。调用前务必查文档或实测 std::chrono 对比耗时。

立即学习C++免费学习笔记(深入)”;

为什么开了 std::execution::par 反而更慢?

并行开销(线程创建/同步/任务划分)远超计算本身时,加速比为负。典型场景包括:

TalkMe
TalkMe

与AI语伴聊天,练习外语口语

下载
  • 输入规模太小(如 std::vector 只有几百个元素)
  • 每项操作极轻量(如 [](int x) { return x * 2; }
  • 迭代器非随机访问(如 std::list 上用 std::for_each)—— 大部分 STL 实现会退化为串行
  • 内存带宽瓶颈(如大量 cache-unfriendly 的随机读写)

建议:对 N 的整型数组,先用串行;超过后才试并行,并用

std::chrono::high_resolution_clock
实测。别依赖理论加速比。

如何正确传递执行策略并避免编译错误

必须显式传入策略对象作为**第一个参数**,且头文件需包含 。常见错误写法:

#include 
#include 

std::vector v(100000, 1); // ❌ 错误:策略位置不对,或漏掉 std::sort(v.begin(), v.end(), std::execution::par);

// ✅ 正确:策略是首个参数,且迭代器类型要匹配 std::sort(std::execution::par, v.begin(), v.end());

// ✅ transform 示例(注意输出迭代器不能和输入重叠,除非用 par_unseq + 无副作用) std::vector out(v.size()); std::transform(std::execution::par_unseq, v.begin(), v.end(), out.begin(), [](int x) { return x * x; });

关键点:策略类型是函数调用的**重载决议依据**,编译器靠它选择并行版本重载。漏传、错位、或用在不支持的算法上,都会 fallback 到串行版本(无声失败),而非报错。

真正容易被忽略的是:并行算法要求迭代器满足 RandomAccessIterator,且谓词/操作必须是无状态、无外部依赖、无副作用(尤其对 par_unseq)。哪怕一个 lambda 捕获了局部 std::mutex,就已违反前提。

相关专题

更多
sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

386

2023.09.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

53

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

204

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

49

2026.01.05

xml格式相关教程
xml格式相关教程

本专题整合了xml格式相关教程汇总,阅读专题下面的文章了解更多详细内容。

0

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.9万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号