互斥锁,信号量,条件变量,临界区

原子操作

共享数据(全局变量或堆变量)的自增(++)操作在多线程环境下会出现错误是因为这个操作(一条c语句)被编译为汇编代码后不止一条指令,因此在执行的时候可能执行了一半就被调度系统打断,去执行别的代码。

我们把单指令的操作称为原子的(Atomic),因为无论如何,单条指令的执行是不会被打断的。为了避免出错,很多体系结构都提供了一些常用操作的原子指令,例如i386就有一条inc指令可以直接增加一个内存单元值。

 

在Windows里,有一套API专门进行一些原子操作(见下表),这些API称为InterlockedAPI

 

Windows API                       作用

 

InterlockedExchange     原子地交换两个值

InterlockedDecrement    原子地减少一个值

InterlockedIncrement    原子地增加一个值

InterlockedXor         原子地进行异或操作

 

使用这些函数时,Windows将保证是原子操作的,因此可以不用担心出现问题。遗憾的是,尽管原子操作指令非常方便,但是它们仅仅适用于比较简单的特定场合。在复杂的场合下,比如我们要保证一个复杂的数据结构更改的原子性,原子操作就力不从心了。这里我们就需要更加通用的手段:锁。

 

同步与锁

为了避免多个线程同时读写同一个数据(全局变量或堆变量)而产生不可预料的后果,我们需要将各个线程对同一个数据的访问同步(Synchronization)。所谓同步,即是指在一个线程访问数据未结束的时候,其他线程不得对同一个数据进行访问。如此,对数据的访问被原子化了。

 

同步的最常见方法是使用锁(Lock)。即每个线程在访问数据或资源之前首先试图获取(Acquire)锁,并在访问结束后释放(Release)锁。在锁已经被占用的时候试图获取锁时,线程会等待,直到锁重新可用。

 

二元信号量(Binary Semaphore)是最简单的一种锁,它只用两种状态:占用与非占用。它适合只能被唯一一个线程访问的资源。当二元信号量处于非占用状态时,第一个试图获取该二元信号量的线程会获得该锁,并将二元信号量置为占用状态,此后其他的所有试图获取该二元信号量的线程将会等待,知道该锁被释放。

 

对于允许多个线程并发访问的资源,多元信号量简称为信号量(Semaphore),它是一个很好的选择。一个初始值为N的信号量允许N个线程并发访问。线程访问资源的时候首先获取信号量,进行如下操作:

■将信号量的值减1

■如果信号量的值小于0,则进入等待状态,否则继续执行。

 

访问资源之后,线程释放信号量,进行如下操作:

■将信号量的值加1.

■如果信号量的值小于1,唤醒一个等待中的线程。

 

互斥量(Mutex)和二元信号量很类似,即资源仅同时允许一个线程访问,但和信号量不同的是,信号量在整个系统可以被任意线程获取并释放,也就是说,同一个信号量可以被系统中的一个线程获取之后由另一个线程释放。而互斥量则要求哪个线程获取了互斥量,哪个线程就要负责释放这个锁,其他线程越俎代庖去释放互斥量是无效的。

 

临界区(Critical Section)是比互斥量更加严格的同步手段。在术语中,把临界区的锁的获取称为进入临界区,而把锁的释放称为离开临界区。临界区和互斥量与信号量的区别在于,互斥量和信号量在系统中任何进程里都是可见的,也就是说,一个进程创建了一个互斥量或信号量,另一个进程试图去获取该锁时合法的。然而,临界区的作用范围仅限于本进程中,其他的进程无法获取该锁(类似于静态全局变量对全局变量)。除此之外,临界区具有和互斥量相同的性质。

 

读写锁(Read-Write Lock)致力于一种更加特定的场合的同步。对于一段数据,多个线程同时读取总是没问题的,但假设操作都不是原子型,只要有任何一个线程试图对这个数据进行修改,就必须使用同步手段来避免出错。如果我们使用上述信号量、互斥量或临界区中的任何一种来进行同步,尽管可以保证程序争取,但对于读取频繁,而仅仅偶尔写入的情况,会显得非常低效。读写锁可以避免这个问题。对于同一个锁,读写锁由两种获取方式,共享的(Shared)独占的(Exclusive)。当锁处于自由的状态时,试图以任何一种方式获取锁都能成功,并将锁置于对应的状态。如果锁处于共享状态,其他线程以共享的方式获取锁仍然会成功,此时这个锁分配给了多个线程。然而,如果其他线程试图以独占的方式获取已经处于共享状态的锁,那么它必须等待锁被所有的线程释放。相应地,处于独占状态的锁将阻止任何其他线程获取该锁,不论它们试图以哪种方式获取。读写锁的行为可以总结为如下表:

 

读写锁状态           以共享方式获取                以独占方式获取

自由                      成功                                  成功

共享                      成功                                  等待

独占                      等待                                  等待

 

条件变量(Condition Variable)作为一种同步手段,作用类似于一个栅栏。对于条件变量,线程可以有两种操作,首先线程可以等待条件变量,一个条件变量可以被多个线程等待。其次,线程可以唤醒条件变量,此时某个或所有等待此条件变量的线程都会被唤醒并继续支持。也就是说,使用条件变量可以让许多线程一起等待某个事件的发生,当事件发生时(条件变量被唤醒),所有的线程可以一起恢复执行。


转自 http://hi.baidu.com/qinfengxiaoyue/item/75c0ddfd13b08f1bfe3582b4



与互斥锁不同,条件变量是用来等待而不是用来上锁的。条件变量用来自动阻塞一个线程,直到某特殊情况发生为止通常条件变量和互斥锁同时使用。

条件变量使我们可以睡眠等待某种条件出现。条件变量是利用线程间共享的全局变量进行同步的一种机制,主要包括两个动作:一个线程等待"条件变量的条件成立"而挂起;另一个线程使"条件成立"(给出条件成立信号)。

条件的检测是在互斥锁的保护下进行的。如果一个条件为假,一个线程自动阻塞,并释放等待状态改变的互斥锁。如果另一个线程改变了条件,它发信号给关联的条件变量,唤醒一个或多个等待它的线程,重新获得互斥锁,重新评价条件。如果两进程共享可读写的内存,条件变量可以被用来实现这两进程间的线程同步。

使用条件变量之前要先进行初始化。可以在单个语句中生成和初始化一个条件变量如:pthread_cond_t my_condition=PTHREAD_COND_INITIALIZER;(用于进程间线程的通信)。可以利用函数pthread_cond_init动态初始化。

条件变量分为两部分: 条件和变量. 条件本身是由互斥量保护的. 线程在改变条件状态前先要锁住互斥量. 它利用线程间共享的全局变量进行同步的一种机制。

相关的函数如下:

1intpthread_cond_init(pthread_cond_t*cond,pthread_condattr_t*cond_attr);   2intpthread_cond_wait(pthread_cond_t*cond,pthread_mutex_t*mutex);3intpthread_cond_timewait(pthread_cond_t*cond,pthread_mutex*mutex,consttimespec*abstime);4 intpthread_cond_destroy(pthread_cond_t*cond);5intpthread_cond_signal(pthread_cond_t*cond);6 intpthread_cond_broadcast(pthread_cond_t*cond);//解除所有线程的阻塞
复制代码
简要说明:     
      (1)初始化.init()或者pthread_cond_t cond=PTHREAD_COND_INITIALIER;属性置为NULL
      (2)等待条件成立.pthread_wait,pthread_timewait.wait()释放锁,并阻塞等待条件变量为真
      timewait()设置等待时间,仍未signal,返回ETIMEOUT(加锁保证只有一个线程wait)
      (3)激活条件变量:pthread_cond_signal,pthread_cond_broadcast(激活所有等待线程)
      (4)清除条件变量:destroy;无线程等待,否则返回EBUSY
 
详细说明

1. 初始化:

    条件变量采用的数据类型是pthread_cond_t, 在使用之前必须要进行初始化, 这包括两种方式:

  • 静态: 可以把常量PTHREAD_COND_INITIALIZER给静态分配的条件变量.
  • 动态: pthread_cond_init函数, 是释放动态条件变量的内存空间之前, 要用pthread_cond_destroy对其进行清理.

#include <pthread.h>int pthread_cond_init(pthread_cond_t *restrict cond, pthread_condattr_t *restrict attr);int pthread_cond_destroy(pthread_cond_t *cond);成功则返回0, 出错则返回错误编号.

    当pthread_cond_init的attr参数为NULL时, 会创建一个默认属性的条件变量; 非默认情况以后讨论.

2. 等待条件:

#include <pthread.h>int pthread_cond_wait(pthread_cond_t *restrict cond, pthread_mutex_t *restric mutex);int pthread_cond_timedwait(pthread_cond_t *restrict cond, pthread_mutex_t *restrict mutex, const struct timespec *restrict timeout);成功则返回0, 出错则返回错误编号.

    这两个函数分别是阻塞等待和超时等待.

    等待条件函数等待条件变为真, 传递给pthread_cond_wait的互斥量对条件进行保护, 调用者把锁住的互斥量传递给函数. 函数把调用线程放到等待条件的线程列表上, 然后对互斥量解锁, 这两个操作是原子的. 这样便关闭了条件检查和线程进入休眠状态等待条件改变这两个操作之间的时间通道, 这样线程就不会错过条件的任何变化.

    当pthread_cond_wait返回时, 互斥量再次被锁住.

3. 通知条件:

#include <pthread.h>int pthread_cond_signal(pthread_cond_t *cond);int pthread_cond_broadcast(pthread_cond_t *cond);成功则返回0, 出错则返回错误编号.

    这两个函数用于通知线程条件已经满足. 调用这两个函数, 也称向线程或条件发送信号. 必须注意, 一定要在改变条件状态以后再给线程发送信号.

示例程序
#include <stdio.h>
#include <pthread.h>
 pthread_mutex_t mutex;
pthread_cond_t cond;
void *thread1(void *arg) 
{
	
pthread_cleanup_push(pthread_mutex_unlock, &mutex);
	
	    //提供函数回调保护
	    while (1) {
		
printf("thread1 is running\n");
		
pthread_mutex_lock(&mutex);
		
pthread_cond_wait(&cond, &mutex);
		
printf("thread1 applied the condition\n");
		
pthread_mutex_unlock(&mutex);
		
sleep(4);
	
}
	
pthread_cleanup_pop(0);

}


void *thread2(void *arg) 
{
	
while (1) {
		
printf("thread2 is running\n");
		
pthread_mutex_lock(&mutex);
		
pthread_cond_wait(&cond, &mutex);
		
printf("thread2 applied the condition\n");
		
pthread_mutex_unlock(&mutex);
		
sleep(1);
	
}

}

int main() 
{
	
pthread_t thid1, thid2;
	
printf("condition variable study!\n");
	
pthread_mutex_init(&mutex, NULL);
	
pthread_cond_init(&cond, NULL);
	
pthread_create(&thid1, NULL, (void *) thread1, NULL);
	
pthread_create(&thid2, NULL, (void *) thread2, NULL);
	
	do {
		
pthread_cond_signal(&cond);
	
} while (1);
	
sleep(20);
	
pthread_exit(0);
	
return 0;

}

 

条件变量与互斥锁、信号量的区别

       1.互斥锁必须总是由给它上锁的线程解锁,信号量的挂出即不必由执行过它的等待操作的同一进程执行。一个线程可以等待某个给定信号灯,而另一个线程可以挂出该信号灯。

       2.互斥锁要么锁住,要么被解开(二值状态,类型二值信号量)。

       3.由于信号量有一个与之关联的状态(它的计数值),信号量挂出操作总是被记住。然而当向一个条件变量发送信号时,如果没有线程等待在该条件变量上,那么该信号将丢失。

       4.互斥锁是为了上锁而设计的,条件变量是为了等待而设计的,信号灯即可用于上锁,也可用于等待,因而可能导致更多的开销和更高的复杂性。


转自http://www.cnblogs.com/newlist/archive/2012/02/11/2346284.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 互斥锁:对于一个写操作,可以使用互斥锁来阻止其他线程读取同一个变量或者资源。读写锁:如果多个线程同时访问一个变量或资源,可以使用读写锁来保证线程安全。条件变量:当某线程需要等待另一个线程完成某任务时,可以使用条件变量来实现。信号量:可以使用信号量来控制同时访问的线程数量,从而保证线程安全。屏障:当多个线程需要在某个时刻同步执行时,可以使用屏障来阻止其他线程继续执行,直到所有线程都到达屏障。 ### 回答2: 互斥锁、读写锁、条件变量信号量、屏障是操作系统中常用的同步原语,用于多线程或多进程之间的协调和同步。下面举几个例子来说明它们的应用: 1. 互斥锁互斥锁是最常见且基础的同步原语,用来保护临界区,确保同一时间只有一个线程可以访问共享资源。例如,在多线程编程中,多个线程需要访问共享的全局变量,我们可以使用互斥锁来保证线程的互斥访问。 2. 读写锁: 读写锁是一种特殊的锁,它分为读锁和写锁。多个线程可以同时获取读锁,但只有一个线程可以获取写锁。读写锁适用于读多写少的场景,可以提高读操作的并发性。例如,在一个文件缓存系统中,多个线程可以同时读取缓存的文件内容,但只有一个线程可以写入缓存。 3. 条件变量条件变量用于线程间的等待和通知机制,能够在满足特定条件时唤醒等待的线程。例如,在生产者-消费者模型中,生产者需要等待缓冲区不满时才能继续生产,消费者需要等待缓冲区不空时才能继续消费。条件变量可以通过等待和通知的方式实现线程的同步。 4. 信号量信号量用于控制对临界资源的访问数量。可以将信号量看作是一个计数器,当资源被占用时,计数器减1,当资源被释放时,计数器加1。例如,在操作系统中,可以使用信号量来限制一个资源的并发访问数量,比如限制同时访问数据库的连接数量。 5. 屏障: 屏障用于控制多个线程在某个点上的同步,即在该点前的线程必须等待所有线程都到达该点才能继续执行。例如,在一个并行计算任务中,可能需要多个线程在某个阶段完成计算后再进入下一个阶段,这时可以使用屏障来同步各个线程的执行。 ### 回答3: 互斥锁是一种保护共享资源的锁,在同一时刻只允许一个线程对共享资源进行操作。例如,当多个线程同时访问一个共享计数器时,互斥锁可以保证每次只有一个线程能够增加或减少计数器的值,避免了竞态条件的发生。 读写锁是一种更高级别的锁,允许多个线程同时读取共享资源,但只允许一个线程进行写操作。例如,在一个并发读写文件的场景中,读写锁可以保证多个线程可以同时读取文件的内容,但只允许一个线程进行写入操作。 条件变量是一种用于线程间通信的机制,可以通知等待的线程某个特定的条件已经满足。例如,在一个生产者-消费者模型中,当缓冲区满时,生产者线程可以通过条件变量通知消费者线程可以消费数据了。 信号量是一种用于控制多个线程并发访问共享资源的机制。例如,当有限数量的资源需要在多个线程间共享时,可以使用信号量来限制资源的并发访问数量,从而避免资源的过度竞争。 屏障用于线程同步,可以让多个线程在某个特定的点上等待,直到所有线程都到达该点才能继续执行。例如,当多个线程并行执行任务,但需要等待所有线程都完成自己的任务后再进行下一步操作时,可以使用屏障来实现线程间的同步。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值