当前位置: 首页
编程语言
C++如何在Linux中实现并发

C++如何在Linux中实现并发

热心网友 时间:2026-05-05
转载

C++在Linux中实现并发的几种核心方法

C++如何在Linux中实现并发

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

想在Linux环境下让C++程序跑得更快、处理能力更强?并发编程是绕不开的关键技术。好消息是,现代C++标准库已经为我们提供了一整套相当完善的并发工具,让开发者能够相对轻松地驾驭多线程、同步等复杂场景。下面就来梳理一下几种最常用、也最核心的实现方式。

1. 多线程:一切并发的起点

说到并发,首先想到的自然是多线程。自C++11起,标准库引入了头文件,创建和管理线程变得前所未有的简单。不再需要依赖平台特定的API,用标准写法就能轻松拉起一个新线程。

#include 
#include 

void helloFunction() {
    std::cout << "Hello from a thread!" << std::endl;
}

int main() {
    std::thread t(helloFunction);
    t.join(); // 等待线程完成
    return 0;
}

看,几行代码就完成了一个线程从创建到执行再到回收的全过程。join()调用确保了主线程会等待工作线程结束,避免了程序提前退出导致的问题。

2. 互斥锁:守护共享资源的卫士

多个线程一旦开始运行,数据共享就成了一个甜蜜的烦恼。如果缺乏保护,对同一块数据的并发读写很容易导致数据竞争,结果往往不可预测。这时候,互斥锁就该登场了。库提供了最基本的锁机制,确保同一时间只有一个线程能进入临界区。

#include 
#include 
#include 

std::mutex mtx; // 创建一个互斥锁

void printMessage(const std::string& msg) {
    mtx.lock(); // 加锁
    std::cout << msg << std::endl;
    mtx.unlock(); // 解锁
}

int main() {
    std::thread t1(printMessage, "Hello from thread 1");
    std::thread t2(printMessage, "Hello from thread 2");
    t1.join();
    t2.join();
    return 0;
}

不过需要留意,直接使用lock()unlock()需要格外小心,万一在加锁后、解锁前发生了异常,可能会导致锁无法释放。更推荐的做法是使用std::lock_guardstd::unique_lock这类RAII风格的包装器,让锁的生命周期自动管理。

3. 条件变量:线程间的“信号灯”

有时候,线程需要等待某个条件成立才能继续执行,而不是盲目地轮询消耗CPU。条件变量就是为解决这类同步问题而生的。它允许一个或多个线程挂起等待,直到另一个线程通知条件发生变化。

#include 
#include 
#include 
#include 

std::mutex mtx;
std::condition_variable cv;
bool ready = false;

void printId(int id) {
    std::unique_lock lck(mtx);
    cv.wait(lck, []{return ready;}); // 等待条件变量
    std::cout << "Thread " << id << std::endl;
}

void go() {
    std::lock_guard lck(mtx);
    ready = true;
    cv.notify_all(); // 通知所有等待的线程
}

int main() {
    std::thread threads[10];
    for (auto& th : threads)
        th = std::thread(printId, &th - &threads[0]);

    std::this_thread::sleep_for(std::chrono::seconds(1)); // 等待一段时间
    go(); // 发出信号

    for (auto& th : threads)
        th.join();
    return 0;
}

这个例子模拟了一个经典的“发令枪”场景:10个线程各就各位,等待一个统一的启动信号。条件变量与互斥锁、谓词(这里的ready标志)配合使用,是构建高效生产者-消费者模型等复杂同步模式的基础。

4. 原子操作:轻量级的并发利器

对于简单的计数器、状态标志这类操作,使用重量级的锁机制可能有些“杀鸡用牛刀”。C++11引入的库提供了原子类型,能够在无需显式加锁的情况下,保证对特定数据的读写操作是原子的、线程安全的。

#include 
#include 
#include 

std::atomic counter(0);

void incrementCounter() {
    for (int i = 0; i < 100000; ++i) {
        counter.fetch_add(1, std::memory_order_relaxed); // 原子加法
    }
}

int main() {
    std::thread t1(incrementCounter);
    std::thread t2(incrementCounter);
    t1.join();
    t2.join();
    std::cout << "Counter: " << counter.load() << std::endl; // 输出计数器的值
    return 0;
}

原子操作的性能通常远高于互斥锁。但要注意,它主要适用于对单个变量的简单操作。对于涉及多个变量需要保持一致性(事务性)的复杂操作,仍然需要依靠锁或其他同步机制。

5. 异步编程:让任务在后台静默执行

有些任务耗时较长,但又不想阻塞主线程。这时候,异步编程模型就派上用场了。允许我们启动一个异步任务,并在未来的某个时刻获取其结果,非常适合于I/O操作、复杂计算等场景。

#include 
#include 

int calculateSum(int a, int b) {
    return a + b;
}

int main() {
    auto future = std::async(std::launch::async, calculateSum, 5, 7); // 异步执行
    std::cout << "Waiting for result..." << std::endl;
    int sum = future.get(); // 获取结果
    std::cout << "The sum is: " << sum << std::endl;
    return 0;
}

std::async的启动策略(std::launch::async)确保了任务会在新线程中执行。主线程通过future.get()获取结果时,如果任务尚未完成,则会自动等待。这种“发起后不管,需要时再取”的模式,能极大地简化并发代码的编写。

6. 信号量:控制并发访问的经典工具

最后提一下信号量。虽然C++标准库至今仍未直接提供信号量,但在Linux平台上,我们可以直接使用POSIX标准定义的信号量(semaphore.h)。信号量本质上是一个计数器,用于控制对有限数量资源的访问,是解决多线程同步问题的又一利器。

#include 
#include 
#include 

sem_t sem;

void workerThread() {
    sem_wait(&sem); // 等待信号量
    std::cout << "Worker thread is processing..." << std::endl;
    sem_post(&sem); // 发送信号量
}

int main() {
    sem_init(&sem, 0, 0); // 初始化信号量,初始值为0
    std::thread t(workerThread);
    std::this_thread::sleep_for(std::chrono::seconds(1)); // 等待一段时间
    sem_post(&sem); // 发送信号量,允许工作线程继续
    t.join();
    sem_destroy(&sem); // 销毁信号量
    return 0;
}

在这个例子中,工作线程在启动后会因信号量初始值为0而阻塞在sem_wait处。主线程在延时后通过sem_post增加信号量,从而“释放”工作线程。这种模式常用于线程池、限流等场景。

以上就是C++在Linux环境下实现并发的几种基本方法。实际开发中,这些技术往往不是孤立使用的。根据具体的性能要求、复杂度和应用场景,灵活地组合运用它们,才能构建出既高效又健壮的并发程序。从简单的多线程到精细的同步控制,现代C++已经为我们提供了足够强大的工具箱,剩下的,就是如何巧妙地运用它们了。

来源:https://www.yisu.com/ask/12588799.html

游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

同类文章
更多
Golang日志对系统资源占用大吗

Golang日志对系统资源占用大吗

总体判断 聊到Go语言日志对系统资源的占用,一个核心结论是:在合理的日志级别和输出策略下,它通常是可控且较小的。不过,事情总有另一面。一旦你遇到高并发、同步落盘、或者低级别日志满天飞的场景,日志就可能摇身一变,成为消耗CPU、内存和I O的“大户”,甚至直接卡住系统的脖子。说到底,影响有多大,关键看

时间:2026-05-05 13:59
Linux系统中Golang日志如何查询

Linux系统中Golang日志如何查询

在Linux系统中查询Golang应用程序日志的实用指南 在Linux环境下,用Golang编写的应用通常会把日志输出到两个地方:要么直接打印在控制台,要么老老实实写进文件里。想找到你需要的日志信息?方法其实就取决于日志去了哪儿。 情况一:日志输出到控制台 这算是最直接的情况了。日志就在终端里滚动,

时间:2026-05-05 13:59
如何在Linux中监控Java日志输出

如何在Linux中监控Java日志输出

在Linux中监控Ja va应用程序的日志输出 处理运行在Linux服务器上的Ja va应用,查看日志是绕不开的日常。面对海量的日志输出,如何高效地捕捉关键信息?其实,系统本身就提供了不少趁手的工具,足以应对大多数场景。下面就来梳理几种常用的方法,你可以根据实际情况灵活选择。 1 使用 `tail

时间:2026-05-05 13:59
strings命令的输出如何保存到文件

strings命令的输出如何保存到文件

将strings命令的输出保存到文件 在处理二进制文件时,strings命令是个非常实用的工具,它能帮助我们提取出文件中的所有可打印字符序列。但很多时候,我们并不满足于仅仅在终端屏幕上扫一眼这些输出,而是需要把它们保存下来,以便后续仔细分析或存档。这该怎么办呢? 其实方法很简单,只需要借助命令行中一

时间:2026-05-05 13:59
strings命令能用于哪些类型的文件

strings命令能用于哪些类型的文件

strings命令:从二进制文件中“打捞”文本的利器 在分析二进制文件时,我们常常需要从一堆机器码中寻找那些人类可读的文本线索。这时,一个名为 strings 的命令行工具就派上了大用场。它堪称是 Unix 和 Linux 系统环境下的“文本打捞器”,专门用于从各类二进制文件中提取出可打印的字符串。

时间:2026-05-05 13:58
热门专题
更多
刀塔传奇破解版无限钻石下载大全 刀塔传奇破解版无限钻石下载大全
洛克王国正式正版手游下载安装大全 洛克王国正式正版手游下载安装大全
思美人手游下载专区 思美人手游下载专区
好玩的阿拉德之怒游戏下载合集 好玩的阿拉德之怒游戏下载合集
不思议迷宫手游下载合集 不思议迷宫手游下载合集
百宝袋汉化组游戏最新合集 百宝袋汉化组游戏最新合集
jsk游戏合集30款游戏大全 jsk游戏合集30款游戏大全
宾果消消消原版下载大全 宾果消消消原版下载大全
  • 日榜
  • 周榜
  • 月榜
热门教程
更多
  • 游戏攻略
  • 安卓教程
  • 苹果教程
  • 电脑教程