竞态条件、数据竞争、缓存变量

java线程与并发编程实践阅读笔记

线程相关的三个问题

  • 竞态条件:当两个线程竞争同一资源,如果对资源的访问顺序敏感,就称存在竞态条件,导致竞态条件发生的代码区称作临界区。

    两个经典例子:
    /** chaeck-then-act :
    	以下代码分为两步(假设a,b为全局变量):
    		a. “检查” 是 if(a == 10.0)
    		b. “动作” 则是 b = a / 2.0 
    竞态条件发生:
    	假设线程1已经执行完了if (a == 10.0),在即将执行b = a / 2/0的时候,被调度器暂停了,此时调度器恢复了另一条线程2 去改变了a的值;当线程1恢复执行的时候,变量b不会等于5.0(此时如果a,b都为局部变量,因为每个线程都会有自己的局部变量拷贝[即变量缓存],所以竞态条件就不会发生)
    */ 
    if(a == 10.0){
        b = a / 2.0;
    }
    
    
    
     /**
     read-moditfy-write:
     counter ++ 是三个单独的操作:
           a.读取counter的值
           b.给值加1
           c.更新值存储到counter中
     
     竞态条件发生情况(依赖于变量缓存):
       线程1调用getId(),此时读取了counter的值,此时值为1。就在此时线程2运行,也调用了getId()方法也读取了值为1,此时线程2 write执行,线程2的counter值为2 ,但是线程1中的counter值仍然为1 线程1走完,线程1 counter的值还是为2。 
    */
    public int getId(){
        return counter ++ ;
    }
    
  • 数据竞争:数据竞争值的是两条或者两条以上的线程并发地访问同一块内存区域,其中至少有一条线程是为了写,而且这些线程没有协调对那块区域的访问。

/**    我开始把数据竞争和竞态条件有点搞混了,竞态条件和数据竞争的区别:
		a.数据竞争主要指的是同一块数据会被多个线程同时操作,被多个线程同时操作时有一个或多个线程写了就是数据竞争。
		b.竟态条件主要指的是当前线程的执行顺序的结果,会被其它线程执行顺序的结果所影响。
		
		数据竞争的例子:单例模式的懒汉式,线程1首先执行了getInstance() 检测到parse是空值,线程1就会去实例化parse并将其赋值Parse。
		当线程2再去执行getInstance() 的时候,此时有两种可能:
			a.检测到parse不为空,直接返回线程1复制的Parse
			b.检测到parse为空,又去创建了一个Parse对象
			(因为线程1和线程2之间没有一个 happens-before ordering[一个动作先于另一个动作发生]的一个保证,所以数据竞争就发生了)		
*/
	private static Parse parse;
	
	public static Perser getInstance()
		{
            if(parse == null){
                parse = new Parse();
            }
            return parse;
		}

  • 缓存变量
	为了提升性能,JVM以及操作系统会协调在寄存器或者处理器中缓存变量,而不是依赖主存。
	每个线程都有自己的变量拷贝,当线程写入这个变量时,其实就是写入自己的拷贝;其他线程使看不到另一个线程的拷贝变量发生改变的。就如竞态条件中的read-moditfy-write例子中线程12中看到的counter都为1一样。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值