目录
信号量
信号量概念
线程间的信号量一般叫匿名信号量(进程间的信号量叫有名信号量),信号量广泛用于进程或线程间的同步和互斥,信号量本质上是一个非负的整数计数器,它被用来控制对公共资源的访问。
编程时可根据操作信号量值的结果判断是否对公共资源具有访问的权限:
当信号量值大于
0
时,则可以访问,否则将阻塞。
PV
原语是对信号量的操作,一次
P
操作使信号量减1,一次
V
操作使信号量加1。
信号量主要用于进程或线程间的同步和互斥这两种典型情况。
信号量用于互斥:
信号量用于同步:
在
POSIX
标准中,信号量分两种,一种是无名信号量,一种是有名信号量。无名信号量一般用于线程间同步或互斥,而有名信号量一般用于进程间同步或互斥。它们的区别和无名管道及有名管道的区别类似,无名信号量则直接保存在内存中,而有名信号量要求创建一个文件。
pthread 库常用的信号量操作函数
pthread 库常用的信号量操作函数
功能:
创建一个信号量并初始化它的值。一个无名信号量在被使用前必须先初始化。
信号量 P 操作(减 1)
阻塞函数
功能:将信号量的值减 1。
操作前,先检查信号量(sem)的值是否为 0,若信号量为 0,此函数会阻塞,直到信号量大于 0 时才进行减 1 操作。
int sem_trywait(sem_t *sem); //非阻塞版本
以非阻塞的方式来对信号量进行减 1 操作。若操作前,信号量的值等于 0,则对信号量的操作失败,函数立即返回。
信号量 V 操作(加 1)
功能:将信号量的值加 1 并发出信号唤醒等待线程(sem_wait())。
获取信号量的值
int sem_getvalue(sem_t *sem, int *sval);
功能:
获取当前
sem
标识的信号量的值,保存在
sval
中。
非
0
表示资源可用
,0
表示资源不可。
销毁信号量
int sem_destroy(sem_t *sem);
功能:
删除
sem
标识的信号量。
无名信号量应用实例
线程
1:
键盘输入字符串 遇到
quit
结束;
线程
2:
读取字符串打印输出;
#include <stdio.h>
#include <unistd.h>
#include <pthread.h>
#include <semaphore.h>
#include <strings.h>
#include <string.h>
sem_t sem_g; //输入进程信号量
sem_t sem_p; //打印进程信号量
char input[1024] = {'\0'};
int ch = '\0';
int i = 0;
//输入进程
void * getchar_thread(void *arg)
{ //线程体:并发执行代码的入口这里开始
while (1) {
sem_wait(&sem_p);
i = 0;
printf("请录入字符串,结束请输入quit:");
while((ch=getchar()) != EOF){
if(ch == '\n'){
continue;
}
input[i] = ch;
i++;
if(strstr(input,"quit") != NULL){
input[i-4] = '\0';
input[i-3] = '\0';
input[i-2] = '\0';
input[i-1] = '\0';
break;
}
}
sem_post(&sem_g);
}
return NULL;
}
//打印进程
void * print_thread(void *arg)
{ //线程体:并发执行代码的入口这里开始
while(1) {
sem_wait(&sem_g);
printf("线程2打印:%s\n",input);
bzero(input, 1024);
sem_post(&sem_p);
}
return NULL;
}
int main(void)
{
pthread_t tid1, tid2;
//初始化信号量
sem_init(&sem_p, 0, 1); //先输入
sem_init(&sem_g, 0, 0); //后打印
//创建2个线程
pthread_create(&tid1, NULL, getchar_thread, NULL);
pthread_create(&tid2, NULL, print_thread, NULL);
//等待线程结束,回收线程资源
pthread_join(tid1, NULL);
pthread_join(tid2, NULL);
return 0;
}