linux代码实操——信号量

信号量描述

信号量是一个特殊的变量,一般取正数值。它的值代表允许访问的资源数目,获取资源时,需要对信号量的值进行原子减一,该操作被称为 P 操作。当信号量值为 0 时,代表没有资源可用,P 操作会阻塞。释放资源时,需要对信号量的值进行原子加一,该操作被称为 V 操作。信号量主要用来同步进程。信号量的值如果只取 0,1,将其称为二值信号量。如果信号量的值大于 1,则称之为计数信号量。
临界资源:同一时刻,只允许被一个进程或线程访问的资源
临界区:访问临界资源的代码段

 

信号量使用

我们结合代码进行操作信号量的接口介绍:
1. #include <sys/sem.h>
2. #include <sys/types.h>
3. #include <sys/ipc.h>
4. /*
5. semget()创建或者获取已存在的信号量
6. semget()成功返回信号量的 ID, 失败返回-1
7. key:两个进程使用相同的 key 值,就可以使用同一个信号量
8. nsems:内核维护的是一个信号量集,在新建信号量时,其指定信号量集中信号
量的个数
9. semflg 可选: IPC_CREAT IPC_EXCL
10. */
11. int semget(key_t key, int nsems, int semflg);
12.
13. /*
14. semop()对信号量进行改变,做 P 操作或者 V 操作
15. semop()成功返回 0,失败返回-1
16. struct sembuf
17. {
18. unsigned short sem_num; //指定信号量集中的信号量下标
19. short sem_op; //其值为-1,代表 P 操作,其值为 1,代表 V 操作
20. short sem_flg; //SEM_UNDO
21. };
22. */
23. int semop(int semid, struct sembuf *sops, unsigned nsops);
24.
25. /*
26. semctl()控制信号量
27. semctl()成功返回 0,失败返回-1
28. cmd 选项: SETVAL IPC_RMID
29.
30. union semun
31. {
32. int val;
33. struct semid_ds *buf;
34. unsigned short *array;
35. struct seminfo *_buf;
36. };
37. */
38. int semctl(int semid, int semnum, int cmd, ...);

例题:进程 a 和进程 b 模拟访问打印机,进程 a 输出第一个字符‘a’表示开始使用打印 机,输出第二个字符‘a’表示结束使用,b 进程操作与 a 进程相同。(由于打印机同一时刻 只能被一个进程使用,所以输出结果不应该出现 abab),如图所示:

 

 

          封装信号量的接口:
sem.h 的代码如下:

 

1. #include <stdio.h>
2. #include <stdlib.h>
3. #include <unistd.h>
4. #include <string.h>
5. #include <assert.h>
6. #include <sys/sem.h>
7.
8. union semun{
9. int val;
10. };
11.
12. void sem_init();//创建/或者已存在的信号量
13. void sem_p();//p 减一
14. void sem_v();//v 加一
15. void sem_destroy();//销毁
sem.c 的代码如下:
1. #include "sem.h"
2.
3. static int semid = -1;
4.
5. void sem_init()//创建/或者已存在的信号量
6. {
7. semid = semget((key_t)1234,1,IPC_CREAT|IPC_EXCL|0600);
8. if ( semid == -1 )//已存在
9. {
10. semid = semget((key_t)1234,1,0600);
11. }
12. else
13. {
14. union semun a;
15.
16. a.val = 1;//信号量的初始值
17. if ( semctl(semid,0,SETVAL,a) == -1 )
18. {
19. perror("semctl error");
20. }
21. }
22. }
23.
24. void sem_p()//p 减一
25. {
26. struct sembuf buf;
27. buf.sem_num = 0;
28. buf.sem_op = -1;//p
29. buf.sem_flg = SEM_UNDO;
30.
31. if ( semop(semid,&buf,1) == -1 )
32. {
33. perror("semop p error");
34. }
35. }
36.
37. void sem_v()//v 加一
38. {
39. struct sembuf buf;
40. buf.sem_num = 0;
41. buf.sem_op = 1;//v
42. buf.sem_flg = SEM_UNDO;
43.
44. if ( semop(semid,&buf,1) == -1 )
45. {
46. perror("semop v error");
47. }
48. }
49.
50. void sem_destroy()//销毁
51. {
52. if ( semctl(semid,0,IPC_RMID) == -1 )
53. {
54. perror("semctl del error");
55. }
56. }
a.c 的代码如下:
1. #include <stdlib.h>
2. #include <unistd.h>
3. #include <string.h>
4. #include <assert.h>
5.
6. int main()
7. {
8. sem_init();
9. int i = 0;
10. for( ;i < 10; i++ )
11. {
12. sem_p();
13. printf("a");
14. fflush(stdout);
15. int n = rand() % 3;
16. sleep(n);
17. printf("a");
18. fflush(stdout);
19. sem_v();
20.
21. n = rand() % 3;
22. sleep(n);
23. }
24. }
b.c 的代码如下:
1. #include <stdlib.h>
2. #include <unistd.h>
3. #include <string.h>
4. #include <assert.h>
5.
6. int main()
7. {
8. sem_init();
9. int i = 0;
10. for( ;i < 10; i++ )
11. {
12. sem_p();
13. printf("b");
14. fflush(stdout);
15. int n = rand() % 3;
16. sleep(n);
17. printf("b");
18. fflush(stdout);
19. sem_v();
20.
21. n = rand() % 3;
22. sleep(n);
23. }
24. }
运行结果如下图所示,输出结果只截了部分:
  • 11
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
信号量是一种用于进程间通信和同步的机制。它是一个计数器,用于保证在共享资源上的互斥访问。在Linux系统中,可以使用信号量来实现进程间的同步和互斥。以下是信号量的基本概念: - 计数器:信号量的值是一个计数器,它可以被多个进程共享。 - P操作:当一个进程需要访问共享资源时,它必须执行P操作,该操作会将信号量的值减1。如果信号量的值为0,则进程将被阻塞,直到信号量的值大于0。 - V操作:当一个进程使用完共享资源后,它必须执行V操作,该操作会将信号量的值加1。如果有进程正在等待该信号量,则唤醒其中一个进程继续执行。 在ZUCC中,可以使用信号量来实现进程的同步和互斥。首先,需要使用semget函数创建一个信号量集合,并使用semctl函数对信号量进行初始化。然后,可以使用semop函数执行P和V操作。例如,下面是一个简单的示例程序,用于演示信号量的使用: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/sem.h> #define SEM_KEY 1234 union semun { int val; struct semid_ds *buf; unsigned short *array; }; int main() { int semid, pid; union semun arg; struct sembuf sb; // 创建信号量集合 semid = semget(SEM_KEY, 1, IPC_CREAT | 0666); if (semid == -1) { perror("semget"); exit(EXIT_FAILURE); } // 初始化信号量 arg.val = 1; if (semctl(semid, 0, SETVAL, arg) == -1) { perror("semctl"); exit(EXIT_FAILURE); } // 创建子进程 pid = fork(); if (pid == -1) { perror("fork"); exit(EXIT_FAILURE); } else if (pid == 0) { // 子进程执行P操作 sb.sem_num = 0; sb.sem_op = -1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop P"); exit(EXIT_FAILURE); } printf("Child process\n"); // 子进程执行V操作 sb.sem_num = 0; sb.sem_op = 1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop V"); exit(EXIT_FAILURE); } exit(EXIT_SUCCESS); } else { // 父进程执行P操作 sb.sem_num = 0; sb.sem_op = -1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop P"); exit(EXIT_FAILURE); } printf("Parent process\n"); // 父进程执行V操作 sb.sem_num = 0; sb.sem_op = 1; sb.sem_flg = SEM_UNDO; if (semop(semid, &sb, 1) == -1) { perror("semop V"); exit(EXIT_FAILURE); } exit(EXIT_SUCCESS); } return 0; } ``` 在上述代码中,创建了一个信号量集合,并将其初始化为1。然后,创建了一个子进程和一个父进程,它们分别执行P和V操作。由于信号量的初始值为1,因此父进程和子进程都可以顺利地执行。如果将信号量的初始值改为0,那么父进程和子进程都将被阻塞,直到有一个进程执行V操作为止。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值