POSIX信号量和SystemV信号量作用相同,都是用于同步操作,达到无冲突的访问共享资源目的。 但POSIX可以用
于线程间同步
初始化信号量
#include <semaphore.h>
int sem_init(sem_t *sem, int pshared, unsigned int value);
参数:
pshared:0表示线程间共享,非零表示进程间共享
value:信号量初始值
销毁信号量
int sem_destroy(sem_t *sem);
等待信号量
功能:等待信号量,会将信号量的值减1
int sem_wait(sem_t *sem);
发布信号量
功能:发布信号量,表示资源使用完毕,可以归还资源了。将信号量值加1。
int sem_post(sem_t *sem);
1.Posix信号量主要完成线程间或进程间的同步与互斥
2.本质
资源计数器+PCB等待队列+提供唤醒和等待接口
资源计数器对临界资源进行计数,信号量通过判断自身资源计数器来进行条件判断。判断当前资源是否可用
可用:对资源进行访问
不可用:进行阻塞等待,直到被唤醒
用循环队列使用Posix信号量
#include <iostream>
#include <unistd.h>
#include <semaphore.h>
#include <pthread.h>
#include <vector>
#define SIZE 1
#define THREADCOUNT 1
class RingQueue
{
public:
RingQueue()
:Vec_(SIZE)
{
CapaCity_ = SIZE;
PosWrite_ = 0;
PosRead_ = 0;
//初始化策略是:信号量计数器的值和数组的空闲空间一样大
sem_init(&ProSem_, 0, SIZE);
//初始化的策略是:在初始化的时候,由于数组没有一个有效元素,所有初始化资源数为0,后边生产线程在唤醒消费线程的时候,会对消费者信号量当中的计数器进行加加操作,从而消费者线程可以获取到消费信号量,进而去访问数组
sem_init(&ConSem_, 0, 0);
//初始化的策略是,初始化资源数为1,即只有一个线程在同一时刻能够拥有该信号量
sem_init(&Lock_, 0, 1);
}
~RingQueue()
{
sem_destroy(&ProSem_);
sem_destroy(&ConSem_);
sem_destroy(&Lock_);
}
void Push(int& Data)
{
sem_wait(&ProSem_);
sem_wait(&Lock_);
Vec_[PosWrite_] = Data;
PosWrite_ = (PosWrite_ + 1) % CapaCity_;
sem_post(&Lock_);
sem_post(&ConSem_);
}
void Pop(int* Data)
{
sem_wait(&ConSem_);
sem_wait(&Lock_);
*Data = Vec_[PosRead_];
PosRead_ = (PosRead_ + 1) % CapaCity_;
sem_post(&Lock_);
sem_post(&ProSem_);
}
private:
std::vector<int> Vec_;
size_t CapaCity_;
//读写位置
int PosWrite_;
int PosRead_;
//同步
sem_t ProSem_;
sem_t ConSem_;
//互斥
sem_t Lock_;
};
void* ConsumeStart(void* arg)
{
RingQueue* rq = (RingQueue*)arg;
int Data;
while(1)
{
rq->Pop(&Data);
printf("ConsumeStart [%p][%d]\n", pthread_self(), Data);
}
return NULL;
}
void* ProductStart(void* arg)
{
RingQueue* rq = (RingQueue*)arg;
int i = 0;
while(1)
{
rq->Push(i);
printf("ProductStart [%p][%d]\n", pthread_self(), i);
i++;
}
return NULL;
}
int main()
{
RingQueue* rq = new RingQueue();
pthread_t com_tid[THREADCOUNT], pro_tid[THREADCOUNT];
int i = 0;
for(; i < THREADCOUNT; i++)
{
int ret = pthread_create(&com_tid[i], NULL, ConsumeStart, (void*)rq);
if(ret < 0)
{
printf("create thread failed\n");
return 0;
}
ret = pthread_create(&pro_tid[i], NULL, ProductStart, (void*)rq);
if(ret < 0)
{
printf("create thread failed\n");
return 0;
}
}
for(i = 0; i < THREADCOUNT; i++)
{
pthread_join(com_tid[i], NULL);
pthread_join(pro_tid[i], NULL);
}
delete rq;
rq = NULL;
return 0;
}