Linux多线程编程五(信号量)

1、什么是信号量数组

      线程的信号量与进程间通讯中使用的信号量的概念是同样,它是一种特殊的变量,它能够被增长或减小,但对其的关键访问被保证是原子操做。若是一个程序中有多个线程试图改变一个信号量的值,系统将保证全部的操做都将依次进行。多线程

      而只有0和1两种取值的信号量叫作二进制信号量,在这里将重点介绍。而信号量通常经常使用于保护一段代码,使其每次只被一个执行线程运行。咱们可使用二进制信号量来完成这个工做。函数


2、信号量接口post


      信号量的函数都以sem_开头,线程中使用的基本信号量函数有4个,它们都声明在头文件semaphore.h中。spa

一、sem_init函数线程

     该函数用于建立信号量,其原型以下:对象

int sem_init(sem_t *sem, int pshared, unsigned int value); 接口

    该函数初始化由sem指向的信号对象,设置它的共享选项,并给它一个初始的整数值。pshared控制信号量的类型,若是其值为0,就表示这个信号量是当前进程的局部信号量,不然信号量就能够在多个进程之间共享,value为sem的初始值。调用成功时返回0,失败返回-1.进程

二、sem_wait函数字符串

       该函数用于以原子操做的方式将信号量的值减1。原子操做就是,若是两个线程企图同时给一个信号量加1或减1,它们之间不会互相干扰。它的原型以下:

int sem_wait(sem_t *sem);  

sem指向的对象是由sem_init调用初始化的信号量。调用成功时返回0,失败返回-1.

函数sem_trywait ( sem_t *sem )是函数sem_wait()的非阻塞版本,它直接将信号量sem的值减一。

三、sem_post函数

该函数用于以原子操做的方式将信号量的值加1。它的原型以下:

int sem_post(sem_t *sem);

与sem_wait同样,sem指向的对象是由sem_init调用初始化的信号量。调用成功时返回0,失败返回-1.

四、sem_destroy函数

该函数用于对用完的信号量的清理。它的原型以下:

int sem_destroy(sem_t *sem);  

成功时返回0,失败时返回-1.

五、获得信号量值

int sem_getvalue (sem_t* sem);


3、使用信号量同步线程

       下面以一个简单的多线程程序来讲明如何使用信号量进行线程同步。在主线程中,咱们建立子线程,并把数组msg做为参数传递给子线程,而后主线程等待直到有文本输入,而后调用sem_post来增长信号量的值,这样就会马上使子线程从sem_wait的等待中返回并开始执行。线程函数在把字符串的小写字母变成大写并统计输入的字符数量以后,它再次调用sem_wait并再次被阻塞,直到主线程再次调用sem_post增长信号量的值。

#include <unistd.h>

#include <pthread.h>

#include <semaphore.h>

#include <stdlib.h>

#include <stdio.h>

#include <string.h>


//线程函数

void *thread_func(void *msg);

sem_t sem;//信号量


#define MSG_SIZE 512


int main()

{

int res = -1;

pthread_t thread;

void *thread_result = NULL;

char msg[MSG_SIZE];

//初始化信号量,其初值为0

res = sem_init(&sem, 0, 0);

if(res == -1)

{

perror("semaphore intitialization failed\n");

exit(EXIT_FAILURE);

}

//建立线程,并把msg做为线程函数的参数

res = pthread_create(&thread, NULL, thread_func, msg);

if(res != 0)

{

perror("pthread_create failed\n");

exit(EXIT_FAILURE);

}

//输入信息,以输入end结束,因为fgets会把回车(\n)也读入,因此判断时就变成了“end\n”

printf("Input some text. Enter 'end'to finish...\n");

while(strcmp("end\n", msg) != 0)

{

fgets(msg, MSG_SIZE, stdin);

//把信号量加1

sem_post(&sem);

}


printf("Waiting for thread to finish...\n");

//等待子线程结束

res = pthread_join(thread, &thread_result);

if(res != 0)

{

perror("pthread_join failed\n");

exit(EXIT_FAILURE);

}

printf("Thread joined\n");

//清理信号量

sem_destroy(&sem);

exit(EXIT_SUCCESS);

}


void* thread_func(void *msg)

{

//把信号量减1

sem_wait(&sem);

char *ptr = msg;

while(strcmp("end\n", msg) != 0)

{

int i = 0;

//把小写字母变成大写

for(; ptr[i] != '\0'; ++i)

{

if(ptr[i] >= 'a' && ptr[i] <= 'z')

{

ptr[i] -= 'a' - 'A';

}

}

printf("You input %d characters\n", i-1);

printf("To Uppercase: %s\n", ptr);

//把信号量减1

sem_wait(&sem);

}

//退出线程

pthread_exit(NULL);

}

相关文章
相关标签/搜索