0

0

C++中如何使用结构化并发_任务调度方案

冰火之心

冰火之心

发布时间:2025-06-22 22:30:02

|

407人浏览过

|

来源于php中文网

原创

c++++结构化并发通过作用域管理任务生命周期,解决资源泄漏和同步问题。1.使用std::jthread自动join线程防止资源泄漏;2.利用std::stop_token安全请求线程停止;3.基于线程池结合std::future和std::packaged_task优化任务调度;4.选择线程池大小时参考cpu核心数与任务类型,通过公式计算并结合性能测试调整;5.避免死锁应确保锁顺序一致、缩短持有时间、设置超时机制;6.避免竞争条件可通过互斥锁、原子操作或无锁数据结构实现。良好的设计与静态分析工具也有助于提升并发安全性。

C++中如何使用结构化并发_任务调度方案

C++结构化并发的核心在于更清晰、可控的任务管理,它通过作用域来管理并发任务的生命周期,避免传统线程管理中常见的资源泄漏和同步问题。任务调度方案则是在此基础上,进一步优化任务的执行顺序和资源分配,提高并发程序的整体效率。

C++中如何使用结构化并发_任务调度方案

解决方案

C++20引入的std::jthreadstd::stop_token是实现结构化并发的关键。std::jthread在线程结束时自动join,防止资源泄漏。std::stop_token则允许安全地请求线程停止,避免强制终止带来的问题。

C++中如何使用结构化并发_任务调度方案

任务调度方案可以基于线程池来实现,结合std::futurestd::packaged_task可以方便地管理任务的执行结果。

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

C++中如何使用结构化并发_任务调度方案

一个简单的示例:

#include 
#include 
#include 
#include 
#include 
#include 
#include 

class ThreadPool {
public:
    ThreadPool(size_t numThreads) : stop(false) {
        for (size_t i = 0; i < numThreads; ++i) {
            workers.emplace_back([this] {
                while (true) {
                    std::function task;

                    {
                        std::unique_lock lock(queueMutex);
                        condition.wait(lock, [this] { return stop || !tasks.empty(); });
                        if (stop && tasks.empty())
                            return;
                        task = std::move(tasks.front());
                        tasks.pop();
                    }

                    task();
                }
            });
        }
    }

    template
    auto enqueue(F&& f, Args&&... args) -> std::future::type> {
        using return_type = typename std::result_of::type;
        auto task = std::make_shared>(
            std::bind(std::forward(f), std::forward(args)...)
        );

        std::future res = task->get_future();
        {
            std::unique_lock lock(queueMutex);
            if (stop)
                throw std::runtime_error("enqueue on stopped ThreadPool");

            tasks.emplace([task]() { (*task)(); });
        }
        condition.notify_one();
        return res;
    }

    ~ThreadPool() {
        {
            std::unique_lock lock(queueMutex);
            stop = true;
        }
        condition.notify_all();
        for (std::thread& worker : workers)
            worker.join();
    }

private:
    std::vector workers;
    std::queue> tasks;
    std::mutex queueMutex;
    std::condition_variable condition;
    bool stop;
};

int main() {
    ThreadPool pool(4);

    std::vector> results;
    for (int i = 0; i < 8; ++i) {
        results.emplace_back(
            pool.enqueue([i]() {
                std::cout << "Task " << i << " executed by thread " << std::this_thread::get_id() << std::endl;
                std::this_thread::sleep_for(std::chrono::seconds(1));
                return i * 2;
            })
        );
    }

    for (auto& result : results) {
        std::cout << "Result: " << result.get() << std::endl;
    }

    return 0;
}

如何选择合适的线程池大小以优化C++并发性能?

线程池大小的选择直接影响并发程序的性能。过小的线程池无法充分利用多核CPU的优势,导致任务排队等待;过大的线程池则可能引入过多的上下文切换开销,反而降低性能。

一个常用的经验法则是:线程池大小 = CPU核心数 * (1 + 等待时间/计算时间)。 其中,等待时间是指任务在等待I/O、网络或其他资源的时间,计算时间是指任务实际执行计算的时间。

例如,如果任务大部分时间都在等待I/O,那么线程池大小可以适当大于CPU核心数。反之,如果任务是CPU密集型的,那么线程池大小接近CPU核心数即可。

mallcloud商城
mallcloud商城

mallcloud商城基于SpringBoot2.x、SpringCloud和SpringCloudAlibaba并采用前后端分离vue的企业级微服务敏捷开发系统架构。并引入组件化的思想实现高内聚低耦合,项目代码简洁注释丰富上手容易,适合学习和企业中使用。真正实现了基于RBAC、jwt和oauth2的无状态统一权限认证的解决方案,面向互联网设计同时适合B端和C端用户,支持CI/CD多环境部署,并提

下载

此外,还可以通过性能测试来确定最佳线程池大小。逐步调整线程池大小,并监控程序的吞吐量、响应时间和CPU利用率,找到一个平衡点。

如何使用std::stop_token优雅地停止C++并发任务?

std::stop_token提供了一种非侵入式的方式来请求线程停止。线程可以通过定期检查std::stop_token::stop_requested()来判断是否需要停止。

#include 
#include 
#include 

void task(std::stop_token stopToken) {
    while (!stopToken.stop_requested()) {
        std::cout << "Task running..." << std::endl;
        std::this_thread::sleep_for(std::chrono::milliseconds(500));
    }
    std::cout << "Task stopped." << std::endl;
}

int main() {
    std::jthread t(task);
    std::this_thread::sleep_for(std::chrono::seconds(3));
    t.request_stop(); // 请求线程停止
    return 0;
}

在这个例子中,task函数会定期检查stopToken.stop_requested(),如果返回true,则停止执行。main函数通过t.request_stop()来请求线程停止。std::jthread会在线程停止后自动join,避免资源泄漏。

需要注意的是,std::stop_token只是提供了一种请求停止的机制,线程需要自行处理停止逻辑。

如何避免C++并发编程中常见的死锁和竞争条件?

死锁和竞争条件是并发编程中常见的难题。死锁是指两个或多个线程互相等待对方释放资源,导致程序无法继续执行。竞争条件是指多个线程同时访问共享资源,导致结果不确定。

避免死锁的常见方法包括:

  • 避免循环等待:确保线程获取锁的顺序一致,避免形成循环依赖。
  • 限制锁的持有时间:尽量减少线程持有锁的时间,避免长时间占用资源。
  • 使用超时机制:在获取锁时设置超时时间,避免无限期等待。

避免竞争条件的常见方法包括:

  • 使用互斥锁:使用std::mutex保护共享资源,确保同一时间只有一个线程可以访问。
  • 使用原子操作:对于简单的共享变量访问,可以使用std::atomic,避免使用锁的开销。
  • 使用无锁数据结构:使用无锁数据结构,例如无锁队列,可以避免锁带来的性能瓶颈。

此外,良好的代码设计和测试也是避免死锁和竞争条件的关键。使用静态分析工具可以帮助发现潜在的并发问题。

相关专题

更多
treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

529

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

4

2025.12.22

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

465

2023.08.10

ip地址修改教程大全
ip地址修改教程大全

本专题整合了ip地址修改教程大全,阅读下面的文章自行寻找合适的解决教程。

86

2025.12.26

压缩文件加密教程汇总
压缩文件加密教程汇总

本专题整合了压缩文件加密教程,阅读专题下面的文章了解更多详细教程。

50

2025.12.26

wifi无ip分配
wifi无ip分配

本专题整合了wifi无ip分配相关教程,阅读专题下面的文章了解更多详细教程。

102

2025.12.26

漫蛙漫画入口网址
漫蛙漫画入口网址

本专题整合了漫蛙入口网址大全,阅读下面的文章领取更多入口。

297

2025.12.26

b站看视频入口合集
b站看视频入口合集

本专题整合了b站哔哩哔哩相关入口合集,阅读下面的文章查看更多入口。

592

2025.12.26

俄罗斯搜索引擎yandex入口汇总
俄罗斯搜索引擎yandex入口汇总

本专题整合了俄罗斯搜索引擎yandex相关入口合集,阅读下面的文章查看更多入口。

729

2025.12.26

热门下载

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

精品课程

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

共94课时 | 5.5万人学习

C 教程
C 教程

共75课时 | 3.7万人学习

C++教程
C++教程

共115课时 | 10.3万人学习

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

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