屏障与互斥量、读写锁等不同,它不是用来保护临界区的,而是与条件变量类似,是一种线程之间的同步机制。屏障允许每个线程等待,直到所有合作线程都到达某一点,然后从该点继续执行。pthread_join就是一种屏障,它允许线程等待直到另一个线程退出。
#include <pthread.h>
pthread_barrier_t ; //屏障数据类型
int pthread_barrier_init(pthread_barrier_t *restrict barrier, const pthread_barrierattr_t *restrict attr, unsigned int count);
//初始化屏障,attr指定屏障属性,NULL为默认属性
//count指定在允许所有线程继续执行之前,必须到达屏障的线程数目
int pthread_barrier_destroy(pthread_barrier_t *barrier);
//撤销屏障
//返回值:成功,返回0 失败则返回错误编码
int pthread_barrier_wait(pthread_barrier_t *barrier);
返回值:若成功,返回0或PTHREAD_BARRIER_SERIAL_THREAD; 否则返回错误编号
调用pthread_barrier_wait的线程在屏障计数未满足条件(计数达到init中登记的数字)时,会进入休眠状态。如果该线程是最后一个调用pthread_barrier_wait的线程,就满足了计数条件,所有线程会被唤醒。第一个返回的线程会返回PTHREAD_BARRIER_SERIAL_THREAD.可以用第一个返回的线程来进行后续的处理工作。
一旦达到屏障计数,该屏障就可以被重用。不过计数值不会改变,除非先调用pthread_barrier_destroy撤销该屏障,之后在调用pthread_barrier_init重新初始化。
下面的程序使用屏障实现一个多线程的排序算法,对于随机生成的15个数分为3部分,使用3个线程分别对各部分进行堆排序。当各部分的排序都完成后(此处用屏障进行各线程的同步),在主线程中对三部分进行归并。对于一个多核处理器来说,这样的多线程排序算法显然优于单线程的算法。对于大量数据来说效率能有显著提高,这里设定总数15个的数据量是为了方便观察结果,只要修改程序中的宏定义即可修改数据总量和使用的线程数。
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <pthread.h>
#define NUMS 15
#define TURN 3
#define N (NUMS/TURN)
pthread_barrier_t the_barrier;
pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
int nums[NUMS];
void adjust(