sem_init:
int sem_init(sem_t *sem, int pshared, unsigned int value);
sem_init函数是Posix信号量操作中的函数。sem_init() 初始化一个定位在 sem 的匿名信号量。value 参数指定信号量的初始值。 pshared 参数指明信号量是由进程内线程共享,还是由进程之间共享。如果 pshared 的值为 0,那么信号量将被进程内的线程共享,并且应该放置在这个进程的所有线程都可见的地址上(如全局变量,或者堆上动态分配的变量)。
参数
sem :指向信号量对象
pshared : 指明信号量的类型。不为0时此信号量在进程间共享,否则只能为当前进程的所有线程共享。
value : 指定信号量值的大小。
sem_post:
int sem_post(sem_t *sem);
sem_post是给信号量的值加上一个“1”,它是一个“原子操作”。
sem_wait:
int sem_wait(sem_t * sem);
sem_wait也是一个原子操作,它的作用是从信号量的值减去一个“1”,但它永远会先等待该信号量为一个非零值才开始做减法。如果对一个值为0的信号量调用sem_wait(),这个函数就会原地等待直到有其它线程增加了这个值使它不再是0为止。如果有两个线程都在sem_wait()中等待同一个信号量变成非零值,那么当它被第三个线程增加 一个“1”时,等待线程中只有一个能够对信号量做减法并继续执行,另一个还将处于等待状态。
简单用例:
typedef struct SocketState
{
int socketNum;
sem_t useFlag;
}SocketState;
int main(void)
{
SocketState svs;
int threadResult = 0;
pthread_t client_id;
sem_init(&svs.useFlag, 0, 1); //初始化信号量,实习同一个进程中线程共享,并且赋值为1.
while(1)
{
sem_wait(&svs.useFlag); // 锁定svs数据,以免在创建的新线程使用前被修改,信号量减一
svs.socketNum = iClientSocketNum;
threadResult = pthread_create(client_id,NULL,(void * (*)(void *))HandleClientConnect,(void *)&svs);//创建一个线程
if(threadResult != 0)
{
sem_post(&svs.useFlag); //信号量加1
}
return 0;
}
static void* HandleClientConnect(SocketState *svs)
{
SocketState mySvs;
mySvs.socketNum = svs->socketNum;
sem_post(&svs->useFlag); //信号量加1
}