文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

深究 Linux 多线程中的信号量 Semaphore

2024-12-03 17:35

关注

 


剑桥词典翻译,并不容易理解

信号:简单来说就是消息,是由用户、系统或者进程发送给目标进程的信息,用来通知目标进程某个状态的改变或系统异常,对应的是异步的场景(我之前的文章有详细介绍过)。

信号量:首先是一个变量,其次是计数器。它是多线程环境下使用的一种设施,信号量在创建时需要设置一个初始值,表示同时可以有几个任务(线程)可以访问某一块共享资源。

另外,对信号量的操作(加、减)都是原子的。互斥锁(Mutex)就是信号量初始值为 1 时的特殊情形,即同时只能有一个任务可以访问共享资源区。

 

Semaphore 再理解

我们来设想这样一个场景(上图):假如北京的国家大剧院有一场免费的音乐会演出,可是现在正值疫情期间,剧院规定:剧院观众总人数要限制,但是允许大家中途退场,把票给其他人,其他人可以中途进场。于是,第一批先到的人从剧院门口票箱中取到了票,然后进场欣赏演出。后到的人就因为剧院满了,在门口等待。过了一段时间,有人嫌节目太无聊了,提前退场了,退场时他把门票放回去了。这样,其他人拿着这个人的票进场了。随后,又有人退场了,但是他忘记把票放回去了。这也没关系,大不了剧院内可容纳的总人数少了一个罢了。

上面的例子中,音乐会现场就是一块共享资源区,观众就是任务(线程),而票箱中的门票数就是信号量。信号量用作并发量限制,由于总的门票数是固定的,所以不会出现音乐厅被挤爆的情况。

上述的例子中,我们允许退场的观众把票带走,这是为什么呢?因为剧院工作人员可以随时在票箱里补充些门票呀(线程生产者)。说到这,你们是不是有点似曾相识呀?对啰,就是线程池,但还是有些不同,你们自己品味吧。

Semaphore 实操练习

信号量类型为 sem_t,类型及相关操作定义在头文件 semaphore.h 中,

创建信号量

  1. int sem_init(sem_t *sem, int pshared, unsigned int value); 

信号量的值加 1

  1. int sem_post(sem_t *sem); 

信号量的值减 1

  1. int sem_wait(sem_t *sem); 

信号量销毁

  1. int sem_destroy(sem_t *sem); 

具体参数含义及返回值,这里就不赘述了。下面展示了一个例子:

你总共有三种类型的下载任务(类型 id 为 1、2、3),每次从键盘读取一种类型的任务进行下载,但是 CPU 最多可以同时执行 2 个下载任务(创建两个线程)。

 

  1. #include  
  2. #include  
  3. #include  
  4. #define MAXNUM (2) 
  5. sem_t semDownload; 
  6. pthread_t a_thread, b_thread, c_thread; 
  7. int g_phreadNum = 1; 
  8.  
  9. void func1(void *arg) 
  10.     // 等待信号量的值 > 0 
  11.     sem_wait(&semDownload); 
  12.     printf("============== Downloading taskType 1 ============== \n"); 
  13.     sleep(5); 
  14.     printf("============== Finished taskType 1 ============== \n"); 
  15.     g_phreadNum--; 
  16.     // 等待线程结束 
  17.     pthread_join(a_thread, NULL); 
  18.  
  19. void func2(void *arg) 
  20.     sem_wait(&semDownload); 
  21.     printf("============== Downloading taskType 2 ============== \n"); 
  22.     sleep(3); 
  23.     printf("============== Finished taskType 2 ============== \n"); 
  24.     g_phreadNum--; 
  25.     pthread_join(b_thread, NULL); 
  26.  
  27. void func3(void *arg) 
  28.     sem_wait(&semDownload); 
  29.     printf("============== Downloading taskType 3 ============== \n"); 
  30.     sleep(1); 
  31.     printf("============== Finished taskType 3 ============== \n"); 
  32.     g_phreadNum--; 
  33.     pthread_join(c_thread, NULL); 
  34.  
  35. int main() 
  36.     // 初始化信号量 
  37.     sem_init(&semDownload, 0, 0); 
  38.     int taskTypeId; 
  39.     while (scanf("%d", &taskTypeId) != EOF) 
  40.     { 
  41.         // 输入 0, 测试程序是否能正常退出 
  42.         if (taskTypeId == 0 && g_phreadNum <= 1) 
  43.         { 
  44.             break; 
  45.         } else if (taskTypeId == 0) 
  46.         { 
  47.             printf("Can not quit, current running thread num is %d\n", g_phreadNum - 1); 
  48.         } 
  49.         printf("your choose Downloading taskType %d\n", taskTypeId); 
  50.         // 线程数超过 2 个则不下载 
  51.         if (g_phreadNum > MAXNUM) 
  52.         { 
  53.             printf("!!! You've reached the max number of threads !!!\n"); 
  54.             continue
  55.         } 
  56.         // 用户选择下载 Task 
  57.         switch (taskTypeId) 
  58.         { 
  59.         case 1: 
  60.             // 创建线程 1 
  61.             pthread_create(&a_thread, NULL, func1, NULL); 
  62.             // 信号量 + 1,进而触发 func1 的任务 
  63.             sem_post(&semDownload); 
  64.             // 总线程数 + 1 
  65.             g_phreadNum++; 
  66.             break; 
  67.         case 2: 
  68.             pthread_create(&b_thread, NULL, func2, NULL); 
  69.             sem_post(&semDownload); 
  70.             g_phreadNum++; 
  71.             break; 
  72.         case 3: 
  73.             pthread_create(&c_thread, NULL, func3, NULL); 
  74.             sem_post(&semDownload); 
  75.             g_phreadNum++; 
  76.             break; 
  77.         default
  78.             printf("!!! error taskTypeId %d !!!\n", taskTypeId); 
  79.             break; 
  80.         } 
  81.     } 
  82.     // 销毁信号量 
  83.     sem_destroy(&semDownload); 
  84.     return 0; 

上述例子中,采用了 pthread_join() 的方式,即子线程合入主线程,主线程阻塞等待子线程结束,然后回收子线程资源。而线程加入还有另外一种方式:pthread_detach(),即主线程与子线程分离,主线程不用关注子线程什么时候结束,子线程结束后,资源自动回收。

程序运行结果如下:

 

还要注意一点:pthread.h 非 linux 系统的默认库, gcc 编译参数需要手动添加选项:-lpthread、-pthread.

来源:今日头条内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯