多线程编程之信号量

     Pthread是 POSIX threads 的简称。是POSIX的线程标准
         相互排斥量用来处理一个共享资源的同步訪问问题,当有多个共享资源时,就须要用到信号量机制。

         信号量机制用于保证两个或多个共享资源被线程协调地同步使用。信号量的值相应当前可用资源的数量。

         1.信号量samaphore):

        信号量机制通过信号量的值控制可用资源的数量。线程訪问共享资源前,须要申请获取一个信号量,假设信号量为0。说明当前无可用的资源,线程无法获取信号量,则该线程会等待其它资源释放信号量(信号量加1)。假设信号量不为0。说明当前有可用的资源,此时线程占用一个资源,相应信号量减1。

        举例:
        停车场有5个停车位,汽车可使用停车位。在这里5个停车位是共享的资源。汽车是线程。開始信号量为5,表明此时有5个停车位可用。一辆汽车进入停车场前。先查询信号量的值,不为0表明有可用停车位,汽车进入停车场并使用一个停车位。信号量减1。表明占用一个停车位,可用数降低。
       
           2.信号量基本函数
         #include <semaphore.h>
初始化信号量:
        int sem_init(sem_t *sem, int pshared, unsigned int val);
        该函数第一个參数为信号量指针,第二个參数为信号量类型(一般设置为0)。第三个为信号量初始值。第二个參数pshared为0时,该进程内全部线程可用。不为0时不同进程间可用。
信号量减1:
        int sem_wait(sem_t *sem);
        该函数申请一个信号量。当前无可用信号量则等待,有可用信号量时占用一个信号量。对信号量的值减1。

信号量加1:
        int sem_post(sem_t *sem);

        该函数释放一个信号量。信号量的值加1。


销毁信号量:
        int sem_destory(sem_t *sem);

        该函数销毁信号量。 
     

        3.牛刀小试
        採用信号量机制,解决苹果橙子问题:一个能放N(这里N设为3)个水果的盘子。爸爸仅仅往盘子里放苹果。妈妈仅仅放橙子。女儿仅仅吃盘子里的橙子,儿子仅仅吃苹果。
        採用三个信号量:
        1.sem_t empty:信号量empty控制盘子可放水果数,初始为3,由于開始盘子为空可放水果数为3。
        2.sem_t  apple ;信号量apple控制儿子可吃的苹果数。初始为0。由于開始盘子里没苹果。
        3.sem_t orange;信号量orange控制女儿可吃的橙子是。初始为0,由于開始盘子里没橙子。

注:相互排斥量work_mutex仅仅为printf输出时可以保持一致,可忽略。
 

#include
#include
#include
#include
#pragma comment(lib, "pthreadVC2.lib") //必须加上这句
sem_t empty; //控制盘子里可放的水果数
sem_t apple; //控制苹果数
sem_t orange; //控制橙子数
pthread_mutex_t work_mutex; //声明相互排斥量work_mutex
void *procf(void *arg) //father线程
{
while(1){
sem_wait(&empty); //占用一个盘子空间,可放水果数减1
pthread_mutex_lock(&work_mutex); //加锁
printf("爸爸放入一个苹果!\n");
sem_post(&apple); //释放一个apple信号了。可吃苹果数加1
pthread_mutex_unlock(&work_mutex); //解锁
Sleep(3000);
} }
void *procm(void *arg) //mother线程
{
while(1){
sem_wait(&empty);
pthread_mutex_lock(&work_mutex); //加锁
printf("妈妈放入一个橙子!\n");
sem_post(&orange);
pthread_mutex_unlock(&work_mutex); //解锁
Sleep(4000);
}
}
void *procs(void *arg) //son线程
{
while(1){
sem_wait(&apple); //占用一个苹果信号量。可吃苹果数减1
pthread_mutex_lock(&work_mutex); //加锁
printf("儿子吃了一个苹果!\n");
sem_post(&empty); //吃了一个苹果,释放一个盘子空间,可放水果数加1
pthread_mutex_unlock(&work_mutex); //解锁
Sleep(1000);
}
}
void *procd(void *arg) //daughter线程
{
while(1){
sem_wait(&orange);
pthread_mutex_lock(&work_mutex); //加锁
printf("女儿吃了一个橙子!\n");
sem_post(&empty);
pthread_mutex_unlock(&work_mutex); //解锁
Sleep(2000);
} } void main()
{
pthread_t father; //定义线程
pthread_t mother;
pthread_t son;
pthread_t daughter; sem_init(&empty, 0, 3); //信号量初始化
sem_init(&apple, 0, 0);
sem_init(&orange, 0, 0);
pthread_mutex_init(&work_mutex, NULL); //初始化相互排斥量 pthread_create(&father,NULL,procf,NULL); //创建线程
pthread_create(&mother,NULL,procm,NULL);
pthread_create(&daughter,NULL,procd,NULL);
pthread_create(&son,NULL,procs,NULL); Sleep(1000000000);
}

【C/C++多线程编程之七】pthread信号量的更多相关文章

  1. 多线程编程(一)-Semaphore(信号量)的使用

    Semaphore的介绍 单词Semaphore的中文含义就是信号.信号系统的意思,此类的主要作用就是限制线程并发的数量. 举个例子,一个屋子里有10个人,但只有一个窄门可以出去,这个窄门一次最多只能 ...

  2. 多线程编程-- part 9 信号量:Semaphore

    Semaphore简介 Semaphore是一个计数信号量,它的本质是一个"共享锁". 信号量维护了一个信号量许可集.线程可以通过调用acquire()来获取信号量的许可:当信号量 ...

  3. C#多线程编程のSemaphore(信号量,负责协调各个线程)

    Semaphore负责协调线程,可以限制对某一资源访问的线程数量 这里对SemaphoreSlim类的用法做一个简单的例子: namespace WpfApplication6 { /// <s ...

  4. 【C/C++多线程编程之十】pthread线程私有数据

    多线程编程之线程私有数据      Pthread是 POSIX threads 的简称.是POSIX的线程标准.         线程同步从相互排斥量[C/C++多线程编程之六]pthread相互排 ...

  5. pthread多线程编程的学习小结

    pthread多线程编程的学习小结  pthread 同步3种方法: 1 mutex 2 条件变量 3 读写锁:支持多个线程同时读,或者一个线程写     程序员必上的开发者服务平台 —— DevSt ...

  6. ZT 为什么pthread_cond_t要和pthread_mutex_t同时使用 || pthread/Linux多线程编程

    为什么线程同步的时候pthread_cond_t要和pthread_mutex_t同时使用 (2009-10-27 11:07:23) 转载▼ 标签: 杂谈 分类: 计算机 举一个例子(http:// ...

  7. 为什么对多线程编程这么怕?pthread,sem,mutex,process

    转自http://blog.chinaunix.net/uid-20788636-id-1841334.html 1.线程创建和退出创建线程实际上就是确定调用该线程函数的入口点,这里通常使用的函数是p ...

  8. Linux多线程编程-信号量

    在Linux中.信号量API有两组.一组是多进程编程中的System V IPC信号量.另外一组是我们要讨论的POSIX信号量. 这两组接口类似,但不保证互换.POSIX信号量函数都已sem_开头,并 ...

  9. Linux下多线程编程-信号量

    今天来谈谈线程的同步--信号量. 首先来看看一些概念性的东西: 如进程.线程同步,可理解为进程或线程A和B一块配合,A执行到一定程度时要依靠B的某个结果,于是停下来,示意B运行:B依言执行,再将结果给 ...

随机推荐

  1. charles-截取移动端请求-设置代理

    Charles 上的设置 1.    要截取 iPhone 上的网络请求,我们首先需要将 Charles 的代理功能打开.在 Charles 的菜单栏上选择 “Proxy”–>“Proxy Se ...

  2. vue 实现邮戳边缘

    效果: vue: <template> <div class="couponItem"> <div class="itemLeft" ...

  3. RuntimeError: You called this URL via POST, but the URL doesn’t end in a slash and you have APPEND_SLASH set.

    做公众号测试的时候,发现了个问题: 提交表单报错:RuntimeError: You called this URL via POST, but the URL doesn’t end in a sl ...

  4. 打包成exe可执行文件的方法

    Python安装扩展库与打包成exe可执行文件的方法 1.安装扩展库的几种方法. 首先可能需要使用 pip install --upgrade pip 来升级本机的pip程序.然后在命令提示符环境(即 ...

  5. [jeecms]获取父栏目下的子栏目名称

    [@cms_channel_list parentId='父栏目id'] [#list tag_list as c] <a href="${c.url}">${c.na ...

  6. volatile和指令重排序

    volatile 的作用 1 精致指令重排序 2 多线程访问同一个变量的时候,每次都是取最新的,而不会使用当前cpu缓存的那一份.

  7. VS2015使用Nuget安装OpenCV3.X以及Python3安装OpenCV3.X

    VS2015已经自带Nuget安装工具了,所以,新建一个项目,点击管理Nuget包 搜索OpenCV3 注意,目前只有这个版本支持VS2015,也就是平台工具集可以为vs140,其他的都会报错,报错我 ...

  8. SQLSTATE[HY000] [2002] 乱码

    string(59) "SQLSTATE[HY000] [2002] ����Ŀ����������ܾ����޷����ӡ� " 实际意思是:SQLSTATE[HY000] [20 ...

  9. 如何将本地项目上传到Git 版本库

    1.(先进入项目文件夹)通过命令 git init 把这个目录变成git可以管理的仓库 git init 2.把文件添加到版本库中,使用命令 git add .添加到暂存区里面去,不要忘记后面的小数点 ...

  10. LUOGU P4171 [JSOI2010]满汉全席

    传送门 解题思路 2-SAT 裸题. 代码 #include<iostream> #include<cstdio> #include<cstring> #inclu ...