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_guard或std::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++已经为我们提供了足够强大的工具箱,剩下的,就是如何巧妙地运用它们了。
相关攻略
Linux环境下C++网络通信:深入解析Socket套接字编程 套接字(Socket)是网络通信的核心端点,它构建了不同计算机间程序数据交换的桥梁。在Linux操作系统中,使用C++实现网络通信主要依赖于Socket编程这套标准化接口。掌握其原理与步骤,是开发高性能网络应用的基础。 本文将详细拆解L
在Linux环境下使用C++实现高效的排序算法 在Linux平台上用C++做开发,排序是绕不开的基础操作。如何实现高效排序?其实路子不少,关键得看场景。下面就来聊聊几种常用的策略和具体实现,从开箱即用的标准库到手动打造的高性能算法,咱们逐一拆解。 1 首选利器:标准库的高效排序函数 绝大多数情况下
Linux下C++容器技术使用指南 一 环境准备与编译运行 要在Linux系统上高效开发基于C++标准模板库(STL)的程序,首要任务是完成开发环境的配置。这一过程的核心在于安装合适的编译器和构建管理工具。其中,GCC G++编译器与CMake构建系统的组合是业界公认的经典方案。 以下是一组可直接执
C++ Linux 平台依赖管理实战指南 一 常用方式与适用场景 在Linux上管理C++依赖,方法不少,各有各的“脾气”和适用场景。选对了,事半功倍;选错了,可能就是一场与编译错误的持久战。 系统级包管理器:这是最“接地气”的方式。在 Debian Ubuntu 系列,你会用 apt 安装像 li
Linux C++网络编程:从基础Socket到现代库的实战指南 想在Linux环境下用C++玩转网络编程?那你来对地方了。这片天地里,从最底层的系统调用到封装完善的高层库,选择其实相当丰富。今天,我们就来聊聊几个最常用、也最值得掌握的网络库,看看它们各自怎么用,又适合哪些场景。 1 Socket
热门专题
热门推荐
争做文明市民:爱护环境卫生从个人点滴做起 本文为您精心汇编了一份全面且实用的“学校创卫宣传标语”大全,旨在为营造更洁净、优美的校园及城市环境提供有力支持。希望这些标语能激发大家的环保热情,共同助力卫生城市创建。更多创卫知识与宣传素材,请持续关注我们的专题栏目。 【学校创卫宣传标语大全】 1、共建卫生
Web3 0底层开发头部项目及对应代币解析 进入2025年,Web3 0的底层开发格局已经相当清晰,一个分层协作的架构体系已然成型。简单来说,基础公链负责提供智能合约的执行环境,跨链协议致力于打通链与链之间的壁垒,存储网络则保障数据的去中心化与持久性,而新兴的开发平台,正以前所未有的方式降低构建门槛
良好的行为习惯是孩子一生发展的基石,而不良习惯则可能阻碍未来的成长道路。一句响亮而清晰的文明礼仪口号,往往比冗长的说教更能触动心灵、凝聚共识,它如同指引方向的灯塔。本文精心整理了适用于小学生的文明礼仪宣传口号,旨在为校园文明建设与德育工作提供实用参考。 小学生文明礼仪口号(1--17条) 1 校园
互联网时代,优秀口号如何赋能品牌与团队凝聚力 在互联网信息蓬勃发展的今天,一句精炼有力的口号,其传播力与影响力不容小觑。优秀的口号不仅能精准传达活动或品牌的核心主题,更能凭借朗朗上口的韵律和深刻的内涵,激发共鸣、凝聚人心。它不仅是团队文化建设的重要基石,也是提升公众认知度的关键载体。您是否正在寻找那
OKX欧易官网:https: www ouzhyi co zh-hans join?channelid=ACE527056&wenzi 说到全球主流的加密货币交易平台,OKX欧易交易所绝对是一个绕不开的名字。它为用户提供了一站式的数字资产服务,从基础的币币交易、合约交易,到资产理财、Web3钱&包





