掌握 C 编程中的多线程:深入讲解和高级概念

掌握 c 编程中的多线程:深入讲解和高级概念

介绍:

C 编程中的多线程使开发人员能够充分利用现代多核处理器的潜力,促进单个进程中任务的并发执行。本综合指南探讨了基本的多线程概念、同步机制和高级主题,为每个概念提供了详细的解释和示例代码。

1. 理解线程:

线程是进程内独立的执行序列,允许并发执行任务。了解线程的创建、管理和状态对于有效的多线程处理至关重要。

线程创建:
pthread_create():初始化一个新线程并开始执行。
pthread_join():等待线程终止后再继续。

#include #include 无效*threadFunc(无效*arg){    printf("来自新线程的你好!n");    pthread_exit(NULL);}int main() {    pthread_t tid;    pthread_create(&tid, NULL, threadFunc, NULL);    pthread_join(tid, NULL);    printf("回到主线程.n");    返回0;}

2、同步与互斥:

当多个线程同时访问共享资源时,会出现竞争条件,从而导致不可预测的行为。互斥体、信号量、条件变量等同步机制保证线程安全。

互斥体(互斥):
互斥体提供互斥,一次只允许一个线程访问共享资源。它们可以防止数据损坏并确保行为一致。

#include #include pthread_mutex_t 互斥体 = PTHREAD_MUTEX_INITIALIZER;int 共享变量 = 0;无效*threadFunc(无效*arg){    pthread_mutex_lock(&mutex);    共享变量++;    printf("线程将共享变量增加到:%dn",共享变量);    pthread_mutex_unlock(&mutex);    pthread_exit(NULL);    }int main() {    pthread_t tid;    pthread_create(&tid, NULL, threadFunc, NULL);    pthread_mutex_lock(&mutex);    共享变量--;    printf("主线程将共享变量递减为:%dn",共享变量);    pthread_mutex_unlock(&mutex);    pthread_join(tid, NULL);    返回0;}

信号量:
信号量是用于控制对共享资源的访问并协调多个线程的执行的同步原语。他们维护一个计数来限制同时访问资源的线程数量。

#include #include #include sem_t 信号量;无效*threadFunc(无效*arg){    sem_wait(&信号量);    printf("线程获取信号量");    // 临界区    sem_post(&信号量);    pthread_exit(NULL);}int main() {    pthread_t tid;    sem_init(&信号量, 0, 1); // 用值 1 初始化信号量    pthread_create(&tid, NULL, threadFunc, NULL);    // 主线程    sem_wait(&信号量);    printf("主线程获取信号量");    // 临界区    sem_post(&信号量);    pthread_join(tid, NULL);    返回0;}

3、线程通信:

线程通信有利于线程之间的协调和同步。条件变量允许线程等待满足特定条件。

条件变量:
条件变量使线程能够等待特定条件的发生。它们通常用于生产者-消费者场景,其中线程在继续之前等待数据可用性。

#include #include pthread_mutex_t 互斥体 = PTHREAD_MUTEX_INITIALIZER;pthread_cond_t condVar = PTHREAD_COND_INITIALIZER;int 数据就绪 = 0;无效*生产者(无效*参数){    pthread_mutex_lock(&mutex);    数据就绪=1;    pthread_cond_signal(&condVar);    pthread_mutex_unlock(&mutex);    pthread_exit(NULL);}无效*消费者(无效*参数){    pthread_mutex_lock(&mutex);    while (!dataReady) {        pthread_cond_wait(&condVar, &mutex);    }    printf("消费者:数据已准备好!n");    pthread_mutex_unlock(&mutex);    pthread_exit(NULL);}int main() {    pthread_t 生产者线程,消费者线程;    pthread_create(&生产者线程,NULL,生产者,NULL);    pthread_create(&consumerThread, NULL, 消费者, NULL);    pthread_join(生产者线程,NULL);    pthread_join(consumerThread, NULL);    返回0;}

4. 先进概念:

优先级反转、饥饿、死锁和自旋锁等高级主题对于构建健壮的多线程应用程序至关重要。

优先级反转:
当低优先级线程持有高优先级线程所需的资源时,就会发生优先级反转,从而导致优先级反转。优先级继承协议通过暂时将低优先级线程的优先级提高到高优先级线程的优先级来帮助缓解此问题。

#include #include pthread_mutex_t mutex1 = PTHREAD_MUTEX_INITIALIZER;pthread_mutex_t mutex2 = PTHREAD_MUTEX_INITIALIZER;无效*highPriorityThread(无效*arg){    pthread_mutex_lock(&mutex1);    pthread_mutex_lock(&mutex2);    // 执行高优先级任务    pthread_mutex_unlock(&mutex2);    pthread_mutex_unlock(&mutex1);    pthread_exit(NULL);}无效*lowPriorityThread(无效*arg){    pthread_mutex_lock(&mutex2);    pthread_mutex_lock(&mutex1);    // 执行低优先级任务    pthread_mutex_unlock(&mutex1);    pthread_mutex_unlock(&mutex2);    pthread_exit(NULL);}int main() {    pthread_t 高优先级,低优先级;    pthread_create(&highPrioTid, NULL, highPriorityThread, NULL);    pthread_create(&lowPrioTid, NULL, lowPriorityThread, NULL);    pthread_join(highPrioTid, NULL);    pthread_join(lowPrioTid, NULL);    返回0;}

饥饿:
当一个线程由于其他线程不断获取所需资源而无法访问这些资源时,就会发生饥饿。公平的调度策略确保所有线程都有公平的资源分配机会,防止饥饿。

#include #include pthread_mutex_t 互斥体 = PTHREAD_MUTEX_INITIALIZER;int 共享资源 = 0;无效*threadFunc(无效*arg){    pthread_mutex_lock(&mutex);    // 增加共享资源    共享资源++;    printf("线程将共享资源增加到:%dn",共享资源);    pthread_mutex_unlock(&mutex);    pthread_exit(NULL);}int main() {    pthread_t tid1, tid2;    // 创建两个线程    pthread_create(&tid1, NULL, threadFunc, NULL);    pthread_create(&tid2, NULL, threadFunc, NULL);    // 等待两个线程完成    pthread_join(tid1, NULL);    pthread_join(tid2, NULL);    // 主线程    pthread_mutex_lock(&mutex);    // 访问共享资源    printf("主线程访问了sharedResource:%dn",sharedResource);    pthread_mutex_unlock(&mutex);    返回0;}

死锁:
当两个或多个线程无限期地等待对方释放它们所需的资源时,就会发生死锁。避免循环等待并实施死锁检测和恢复机制有助于缓解死锁情况。

#include #include pthread_mutex_t mutex1 = PTHREAD_MUTEX_INITIALIZER;pthread_mutex_t mutex2 = PTHREAD_MUTEX_INITIALIZER;无效*线程1(无效*参数){    pthread_mutex_lock(&mutex1);    pthread_mutex_lock(&mutex2);    // 临界区    pthread_mutex_unlock(&mutex2);    pthread_mutex_unlock(&mutex1);    pthread_exit(NULL);}无效*线程2(无效*参数){    pthread_mutex_lock(&mutex2);    pthread_mutex_lock(&mutex1);  // 潜在的死锁点    // 临界区    pthread_mutex_unlock(&mutex1);    pthread_mutex_unlock(&mutex2);    pthread_exit(NULL);}int main() {    pthread_t tid1,tid2;    pthread_create(&tid1, NULL, thread1, NULL);    pthread_create(&tid2, NULL, thread2, NULL);    pthread_join(tid1, NULL);    pthread_join(tid2, NULL);    返回0;}

自旋锁:
自旋锁是同步原语,其中线程不断轮询资源的可用性。它们对于短关键部分和低争用场景非常有效。

#include #include pthread_spinlock_t 自旋锁;无效*threadFunc(无效*arg){    pthread_spin_lock(&spinlock);    // 临界区    printf("线程获得了自旋锁n");    // 执行一些任务    pthread_spin_unlock(&spinlock);    pthread_exit(NULL);}int main() {    pthread_t tid1, tid2;    pthread_spin_init(&spinlock, 0);    pthread_create(&tid1, NULL, threadFunc, NULL);    pthread_create(&tid2, NULL, threadFunc, NULL);    pthread_join(tid1, NULL);    pthread_join(tid2, NULL);    pthread_spin_destroy(&spinlock);    返回0;}

结论:

掌握 C 编程中的多线程需要深入理解基本概念、同步机制和高级主题。通过深入研究这些概念并探索示例代码,开发人员可以构建健壮、高效且响应迅速的多线程应用程序。持续实践、实验和遵守最佳实践是精通多线程和开发充分利用现代硬件功能的可靠软件系统的关键。

以上就是掌握 C 编程中的多线程:深入讲解和高级概念的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1455331.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月18日 07:40:22
下一篇 2025年12月18日 07:40:38

相关推荐

  • Linux AnkiDroid同步,HTML+CSS移动学习无缝!

    首先确保Anki桌面端与AnkiWeb成功同步,再在Android设备安装AnkiDroid并登录同一账户,接着通过编辑卡片模板添加响应式HTML与CSS代码以适配移动端,然后测试样式在AnkiDroid中的渲染效果并调整不兼容属性,最后启用双向同步机制,确保移动与桌面端修改可互相更新,实现跨平台无…

    2025年12月23日
    000
  • HTML表单数据怎么同步更新_HTML多个表单元素数据同步更新的实现方法

    使用JavaScript监听输入事件可实现表单数据同步,如通过input事件实时更新多个输入框或文本元素;还可借助公共变量管理状态,简化多元素同步逻辑;对于复杂场景,推荐使用Vue等框架的双向绑定,自动保持数据一致。 在前端开发中,多个表单元素之间数据同步更新是一个常见需求。比如用户在一个输入框填写…

    2025年12月23日
    000
  • OneDrive跨设备同步,HTML+CSS走到哪写到哪!

    OneDrive通过云同步实现HTML和CSS代码跨设备实时协作。将项目存于OneDrive文件夹并登录账户,可自动同步至所有设备;在Surface Pro 9运行Windows 11环境下,使用Visual Studio Code打开OneDrive中的项目目录,保存即触发后台同步;移动端安装On…

    2025年12月23日
    000
  • Mac iCloud Drive自动备份每天HTML练习代码

    首先启用iCloud Drive并登录Apple ID,将HTML代码文件夹移入iCloud Drive目录,每天创建日期命名的子文件夹分类存储,通过桌面别名快速访问,并定期检查文件同步状态以确保备份成功。 如果您希望在Mac上使用iCloud Drive自动备份每天编写的HTML练习代码,可以通过…

    2025年12月23日
    000
  • DataTables列可见性与搜索框同步控制教程

    本文旨在解决datatables表格中,当动态显示或隐藏列时,其对应的列搜索输入框未能同步隐藏或显示的问题。核心在于理解datatables的dom结构和列可见性api的工作原理,并提供两种解决方案:优化dom结构将搜索框与列头紧密关联,或通过手动同步机制确保列搜索框与列的可见性保持一致。 在构建交…

    2025年12月23日
    000
  • HTML表单如何实现协作编辑?怎样多人同时编辑同一表单?

    html表单本身不支持多人协作编辑,必须通过websocket实现实时通信,并结合后端协调与前端响应机制,利用操作转换(ot)或crdts等算法处理并发冲突,最终在前端通过javascript监听并更新表单状态,实现多人实时协同编辑,且可通过视觉反馈增强协作体验。 HTML表单实现多人同时协作编辑,…

    2025年12月22日
    000
  • HTML表单如何防止重复提交?提交后如何禁用提交按钮?

    答案:防止表单重复提交需前后端结合,前端通过禁用按钮和提交状态标志提供即时反馈,后端则利用令牌机制、幂等性键、数据库唯一约束及业务状态校验确保数据安全,二者协同实现用户体验与系统可靠性的平衡。 防止HTML表单重复提交,通常需要在客户端和服务器端双管齐下,而提交后禁用按钮则是客户端最直观、用户体验最…

    2025年12月22日
    000
  • HTML表单如何实现数据绑定?怎样自动填充表单字段?

    答案:数据绑定通过事件监听实现表单与数据模型的实时同步,自动填充则通过HTML属性、JavaScript或浏览器功能预设表单值;二者协同工作但关注点不同,前者强调双向同步,后者侧重初始便捷性。 HTML表单实现数据绑定,本质上是将表单控件的值与JavaScript中的某个数据模型(比如一个对象)同步…

    2025年12月22日
    000
  • HTML表单如何实现PWA支持?怎样添加离线功能?

    答案是利用Service Worker缓存资源并结合Background Sync API实现离线提交与自动同步。通过注册Service Worker缓存表单相关文件,拦截提交行为,将离线数据存入IndexedDB,并注册后台同步任务,待网络恢复后由Service Worker自动发送数据,确保提交…

    2025年12月22日
    000
  • HTML表单如何实现断网检测?怎样在离线时保存表单数据?

    答案:通过navigator.onLine和online/offline事件检测网络状态,结合localStorage或IndexedDB离线存储表单数据,网络恢复后触发同步机制,利用fetch发送数据并实现幂等性处理,确保数据安全可靠提交。 HTML表单在断网时进行检测并保存数据,核心在于利用浏览…

    2025年12月22日
    000
  • HTML表单如何实现故障转移?怎样处理服务器宕机?

    客户端可通过localstorage实时缓存表单数据并在页面加载时恢复,提交成功后清除缓存,以防止用户输入丢失,对于敏感数据应避免使用此方式或结合加密处理,此方法能有效提升用户体验并保障数据完整性。 HTML表单本身并没有内置的“故障转移”机制,它更像是一个数据提交的入口。当谈到表单的故障转移和服务…

    2025年12月22日
    000
  • HTML表单如何实现CRM集成?怎样同步数据到销售系统?

    答案:HTML表单通过后端服务器将数据发送至CRM API,经验证、映射和认证后实现集成,再通过Webhook或第三方平台将数据同步至销售系统。主要技术路径包括后端直连API、前端直连(不推荐)、CRM嵌入式表单和iPaaS平台。安全性需依赖HTTPS、后端验证、CSRF防护和敏感信息隔离,准确性则…

    2025年12月22日
    000
  • HTML表单如何实现代码编辑器?怎样高亮显示代码语法?

    要在HTML表单中实现代码编辑器并高亮显示代码语法,核心是使用JavaScript库如CodeMirror或Monaco Editor将普通textarea或div升级为功能完整的编辑器,通过引入库文件、语言模式和主题,初始化编辑器实例,并在表单提交前将编辑器内容同步回隐藏的textarea或inp…

    2025年12月22日
    000
  • HTML5的Web Workers是什么?如何实现多线程?

    web workers对前端开发至关重要,因为它允许javascript在后台线程中执行耗时任务而不阻塞主线程,从而提升页面响应性和用户体验。1. web workers通过创建独立线程处理计算密集型任务,如数据处理、图像操作和复杂算法;2. 主线程与worker之间通过postmessage和on…

    2025年12月22日 好文分享
    000
  • 如何用JavaScript和接口时间戳实现精准的秒级倒计时?

    利用JavaScript和服务器时间戳构建精准秒级倒计时器 许多应用场景,例如限时抢购、活动倒计时等,都需要一个精确到秒的倒计时功能。本文将演示如何使用javascript结合服务器返回的时间戳,实现一个实时更新的秒级倒计时器。 核心思路是:持续获取客户端当前时间戳,并与服务器返回的初始时间戳进行比…

    2025年12月22日
    000
  • IE浏览器弹窗句柄查找失败:为什么父窗口无法枚举其子窗口弹窗句柄?

    ie浏览器自动化难题:父窗口无法枚举子窗口弹窗句柄 在使用自动化工具操控IE浏览器时,弹窗处理是常见挑战。本文分析一个用户遇到的IE弹窗句柄查找问题:用户先获取弹窗句柄,再获取其父窗口(IE浏览器窗口)句柄。但使用父窗口句柄枚举子窗口时,却找不到之前获取的弹窗句柄,尽管弹窗仍然可见。 此问题可能由以…

    2025年12月22日
    000
  • IE弹窗句柄和父窗口句柄:为什么枚举父窗口子句柄找不到弹窗句柄?

    ie弹窗句柄与父窗口句柄:查找失败原因分析 在自动化测试或脚本中,获取IE弹窗句柄及其父窗口句柄至关重要。然而,实际操作中常遇到一个难题:已获取弹窗句柄和父窗口句柄,但通过枚举父窗口子句柄却无法找到该弹窗。本文将分析可能导致此问题的原因。 问题描述: 已获取IE弹窗句柄,并确认其父窗口句柄。但使用父…

    2025年12月22日
    000
  • React父子组件引用错位的原因是什么?

    react父子组件引用错位详解及排查 在React开发中,有时会遇到父组件引用与实际渲染的子组件不符的情况,即父子组件引用错位。本文将分析此问题产生的常见原因,并提供相应的解决方法。 可能原因: 状态管理不当:当子组件依赖父组件状态时,若状态提升或状态同步机制存在问题,可能导致子组件渲染错误,从而出…

    2025年12月22日
    000
  • React中父子组件数据传递与状态同步:实现子组件操作父组件列表数据更新

    本文详细探讨react中父子组件间的数据传递与状态同步机制,重点解决子组件修改父组件数组数据并更新视图的问题。文章通过两种主要方法——传递父组件的状态设置函数和传递特定操作的回调函数——演示了如何实现子组件删除列表项后,父组件状态随之更新,并强调了组件间职责分离的最佳实践。 引言:React组件通信…

    2025年12月21日 好文分享
    000
  • 解决页面刷新后暗模式切换图标不同步的问题

    本教程详细讲解如何确保网页的暗模式切换图标在页面刷新后依然能正确反映当前的暗模式状态。通过分析原始代码的问题,我们将展示如何利用 `localstorage` 存储的状态,在页面加载时同步更新图标的显示,从而提供一致的用户体验。核心在于修改切换函数以同时管理图标可见性,并在页面初始化时根据存储状态调…

    2025年12月21日
    000

发表回复

登录后才能评论
关注微信