条件变量的工作原理:对当前不访问共享资源的任务,直接执行睡眠处理,如果此时需要某个任务访问资源,直接将该任务唤醒。条件变量类似异步通信,操作的核心:睡眠、唤醒。

1.pthread_cond_t  定义条件变量

2.初始化条件变量

  函数原型:int pthread_cond_init(pthread_cond_t *restrict cond,const pthread_condattr_t *restrict attr);

  功能  :初始化条件变量

  参数:cond   条件变量标识符

      addr  属性

  返回值:成功  :0;失败:错误码

3.休眠

  函数原型:int pthread_cond_wait(pthread_cond_t *restrict cond,pthread_mutex_t *restrict mutex);

  功能:  任务执行睡眠。执行睡眠之前,必须执行上锁

  参数:  cond  条件变量标识符

        mutex  互斥锁标识符

  返回值:  成功: 0;失败:错误码

4.执行唤醒操作

  函数原型:int pthrerad_cond_signal(pthread_cond_t *cond);

  功能:执行唤醒操作

  参数:cond  条件变量标识符

  返回值: 成功 0;失败:错误码

5.摧毁条件变量

  函数原型:int pthread_cond_destroy(pthread_cond_t *cond);

  功能:  摧毁条件变量

  参数:  cond  条件变量标识符

  返回值:成功:0;失败 :错误码。

注意:在睡眠之前,执行上锁处理,将睡眠操作锁住,在线程执行睡眠的时候,不会被打断。

    pthread_mutex_lock(&lock);

   线程睡眠,函数阻塞,函数  自动  释放锁资源,此时线程得到唤醒信号,pthread_cond_wait立刻返回,有阻塞编程非阻塞,pthread_cond_wait立刻自动执行上锁处理

    pthread_cond_wait(&cond,&lock);

    注意:   唤醒操作发生在睡眠之前,则唤醒操作无效。所以,发送信号一定要咋睡眠之后。

  printf("buf=%s\n",buf);      临界区

  pthread_mutex_unlock(&lock);  解锁处理

无名管道:

  1.要点:

    1)只能用于具有亲缘关系的进程之间的通信

    2)半双工的通信方式,具有固定的读端和写端

    3)管道可以看成是一种特殊的文件,对于它的读写可以使用文件IO如read/write函数

    4)无名管道的操作属于一次性操作,如果对无名管道执行读操作,数据就会被读走(类似剪切的方式)

    5)无名管道管道的大小是固定的,管道一旦写满,写操作就会阻塞,管道的大小为64K

    6)当管道中无数据,执行读操作,读操作阻塞

    7)无名管道写满,写操作阻塞,如果管道中有大于4k的空间,写操作就可以继续,每次最多写入4k的整数倍

    8)无名管道不保证操作的原子性,如果当前管道,满足读写条件,读写可以并发。

    9)向无名管道中写数据,将读端关闭,管道损坏,进程收到信号(SIGPIPE)将进程退出

    10)当管道中有数据,将写端关闭,读操作可以执行,之后数据读完,可以继续读取(非阻塞)

  2.创建无名管道:

    函数原型:int pipe(int pipefd[2]);

    功能  :创建无名管道

    参数:  pipefd[2]  保存的是操作管道的两个文件描述符

              pipefd[1]管道的写端

              pipefd[0]管道的读端

    返回值:成功 0;失败:-1;

有名管道:

1.要点

  1)有名管道可以使互不相关的两个进程互相通信。

  2)有名管道可以通过路径名来指出,并且在文件系统中可见

        (文件系统:提供文件维护的机制)

  3)进程通过文件IO来操作有名管道

  4)有名管道遵循先进先出规则

  5)不支持如LSEEK()操作

  6)有名管道:先创建,再打开。

2.相关函数  

  1)创建有名管道

    函数原型:int mkfifo(const char *pathname,mode_t mode);

    功能  :创建有名管道

    参数  :pathname  路径名(管道名)

          mode   文件所属用户的执行权限

    返回值:成功:0 ;失败 -1

信号:

  1.要点

    信号是一种异步通信机制,信号可以直接进行用户空间进程和内核进程之间的交互

    内核进程也可以利用它来通知用户空间进程发生了那些系统事件

    用户进程对信号的相应方式:

      1.忽略信号:对信号不做任何处理,但是有两个信号不能忽略:SIGKILL和 SIGSTOP

      2捕捉信号:用户自定义信号处理函数,当信号发生时,执行相应的处理函数。

      3.执行默认操作:Linux对每种信号都规定了默认操作

      可用   kill   -l   查看当前linux可用的信号

  2.相关函数

    1)注册信号

        函数原型:type void (*sighandler_t(int));  == typedef void(*)sighandler_t;

            sighandler_t   signal(int signum,sighandler_t handler);

        功能   :注册信号

        参数  :signum  注册信号的类型

             handler  对注册信号的处理,自定义函数:执行捕捉信号

                              SIG_IGN  忽略信号

                              SIG_DFL  执行信号的默认操作

        返回值:失败:SIG_ERR

    2)想进城或进程组发送任何一个信号

      函数原型:int kill(pid_t pid,int sig);

       功能  :向进程或进程组发送任何一个信号

        参数   :pid  》0  指定进程号进行发送

              = 0 信号发送给与调用进程在同一个进程组下的所有进程

              -1 信号发送给当前进程允许发送的任何一个进程(init进程)

              <-1 信号发送个进程组ID等于-PID下的任何一个进程

            sig  发送信号的类型

    3)给调用进程发送信号

        函数原型:int raise(int sig);

            功能:给调用进程发送信号

            参数:sig  信号的类型

    4)函数时钟设置

      函数原型:unsigned int alarm(unsigned int seconds);

      参数:  sec 秒数(从设置时钟开始,到sec秒之后,进程收到信号SIGALRM,进程退出)

      返回值:  成功  0

    

  在alarm 之后,在次执行alarm,新的时钟将会替换上一次时钟设置,开始以新的时钟开始计时

  会返回上一次执行ALARM到当前执行ALARM剩余的时间。

posted on 2018-07-31 20:05  子都  阅读(143)  评论(0编辑  收藏  举报