并发编程笔记五:活跃性问题

五、活跃性问题

活跃性问题包括死锁、活锁、饥饿,最著名的就是死锁问题了。

5.1、死锁

5.1.1、什么是死锁

系统中的每一个线程都持有着其他线程需要的资源,同时又想获取其他线程已经拥有的资源,并且每个线程在获取全部想要的资源之前不会释放已经拥有的资源,那么这种情况就称为死锁。

例如:当线程一持有A锁,想要获取B锁的同时,线程二持有者B锁,并且想要获取A锁。那么线程一和线程二就产生了死锁,两个线程会永远的阻塞下去。

5.1.2、死锁代码

public class DeadLockDemo {
	public static void main(String[] args) throws Exception{
	     Object lock1 = new Object();
	     Object lock2 = new Object();
	     Thread t1 = new Thread(() -> {
	         synchronized (lock1) {
	             System.out.println("t1获取lock1");
	             try {
	                 Thread.sleep(1000);
	             } catch (InterruptedException e) {
	                 e.printStackTrace();
	             }
	             synchronized (lock2) {
	                 System.out.println("t1获取lock2");
	             }
	         }
	     });
	     Thread t2 = new Thread(() -> {
	         synchronized (lock2) {
	             System.out.println("t2获取lock2");
	             try {
	                 Thread.sleep(1000);
	             } catch (InterruptedException e) {
	                 e.printStackTrace();
	             }
	             synchronized (lock1) {
	                 System.out.println("t2获取lock1");
	             }
	         }
	     });
	     t1.start();
	     t2.start();
	     t1.join();
	     t2.join();
	     System.out.println("执行完毕");
	}
}

 

执行结果:

t1获取lock1

t2获取lock2

5.1.3、如何辨别死锁代码

(1)辨明代码中所有的同步代码共有几个锁对象,如果只有一个,是不会产生死锁的,如果有多个则需要小心;

(2)在同步代码块中,是否有调用其他方法的情况,如果有就有很大的风险。

(3)查看这些同步代码块调用其他同步方法的代码的获取锁的顺序,如果多个这种调用的情况顺序是不一样的,那么就有可能产生死锁。

5.1.4、死锁的必要条件

(1)互斥条件:一个资源一次只能有一个线程可以使用,如果有另一个线程申请该资源,那么这个线程进入阻塞状态知道资源被释放;(synchronized独占锁)

(2)请求与保持条件:一个线程因请求了被其他线程占用的资源而进入阻塞状态时,这个线程对自己持有的资源不会释放;

(3)不剥夺条件:一个线程获取了某个资源后,只能进程自己资源释放资源,不能强行剥夺;

(4)循环等待条件:有一组等待线程 {P0,P1,…,Pn},P0 等待的资源为 P1 占有,P1 等待的资源为 P2 占有,……,Pn-1 等待的资源为 Pn 占有,Pn 等待的资源为 P0 占有。

这四个条件是形成死锁的必要条件,非充分条件,即同时满足这四个条件也不一定会产生死锁,但是产生了死锁,肯定是满足了这四个条件。只有上诉四个条件之一不满足,肯定不会产生死锁。

5.1.5、如何避免死锁

(1)开放调用

不要在持有锁的情况下调用其他方法,所以要尽量缩小锁的范围,只对共享可变变量的操作上锁即可。

这种方法是避免了循环等待条件的形成。

(2)固定获取资源的顺序

对于线程获取资源的顺序进行限制,可以对资源进行大小排序,获取资源时要从最大的资源开始获取。

这种方法也是避免了循环等待条件的形成。

(3)使用超时锁

超时锁可以在锁使用一段时间后,自动释放锁,详见?显示锁

这种方法是打破了不剥夺条件。

 

5.2、饥饿

由于进程的优先级过低而一直获取不到cpu的执行权。这种问题称为饥饿。

所以我们尽量不要设置进程的优先级,都是用默认优先级即可。

5.3、活锁

当多个相互协作的线程都对彼此进行响应从而修改各自的状态,并使得任何一个线程都无法继续执行,就发生了活锁。

避免活锁的方法是在重试机制中加入随机性。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
嗨!很高兴回答你关于Java并发编程问题。请问你想知道什么方面的内容呢?我可以分享一些学习笔记和建议给你。 1. 并发编程基础:了解并发编程的基本概念,如线程、进程、锁、同步等。学习Java中的并发编程模型以及相关的API,如Thread、Runnable、Lock、Condition等。 2. 线程安全性:学习如何保证多线程环境下的数据安全性,了解共享资源的问题以及如何使用同步机制来防止数据竞争和并发问题。 3. 线程间的通信:掌握线程间的通信方式,如使用wait/notify机制、Lock/Condition等来实现线程的协调与通信。 4. 并发容器:学习并发容器的使用,如ConcurrentHashMap、ConcurrentLinkedQueue等。了解它们的实现原理以及在多线程环境下的性能特点。 5. 并发工具类:熟悉Java提供的并发工具类,如CountDownLatch、CyclicBarrier、Semaphore等,它们可以帮助你更方便地实现线程间的协作。 6. 并发编程模式:学习一些常见的并发编程模式,如生产者-消费者模式、读者-写者模式、线程池模式等。了解这些模式的应用场景和实现方式。 7. 性能优化与调试:学习如何分析和调试多线程程序的性能问题,了解一些性能优化的技巧和工具,如使用线程池、减少锁竞争、避免死锁等。 这些只是一些基本的学习笔记和建议,Java并发编程是一个庞大而复杂的领域,需要不断的实践和深入学习才能掌握。希望对你有所帮助!如果你有更具体的问题,欢迎继续提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值