Posix信号量

    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;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值