java并发编程学习之死锁

  • 死锁的介绍:
    锁是一个非常有用的工具,运行场景非常多,因为它使用起来非常简单,而且易于理解。但同时它也会带来一些困扰,比如死锁问题。比如有两个线程A和B,运行都需要两个资源a,b。A获取了a资源,B获取了b资源,接着A去请求b资源,B去请求a资源,两个线程互相阻塞产生死锁。

  • 代码示例:

public calss DeadLockDemo{
	private static String A = "A";
	private static String B = "B";
	public static void main(String[] args){
		new DeadLockdemo().deadLock();
	}
	private void deadLock(){
		Thread t1 = new Thread(new Runnable(){
			@Override
			public void run(){
				synchronized(A){
					try{
						Thread.currentThread().sleep(2000);
					}catch(Exception e){
					  e.printStackTrace();
					 }
					 synchronized(B){
					 	System.out.println("B");
					 }
				}
			}
		});
		Thread t2 = new Thread(new Runnable(){
			@Override
			public void run(){
				synchronized(B){
					try{
						Thread.currentThread().sleep(2000);
					}catch(Exception e){
					  e.printStackTrace();
					 }
					 synchronized(A){
					 	System.out.println("A");
					 }
				}
			}
		});
		t1.start();
		t2.start();
	}
}

上述代码执行后会发生死锁,t1和t2互相阻塞。

  • 死锁产生的场景分析:
    在一个更为复杂的场景中,你可能会遇到这样的问题,t1拿到锁之后,因为一些异常情况而没有释放锁(死循环)。或者t1拿到了一个数据库锁,释放锁的时候抛出了异常没释放掉。

  • 避免死锁的几种方法:
    1.尽量避免一个线程同时获取多个锁。
    2.尽量避免一个线程同时占据多个支援,尽量一个线程同时只占据一个资源。
    3.尝试使用定时锁。Lock.tryLock(timeout)来代替使用内部锁机制。
    4.对于数据库锁,加锁和解锁必须在一个数据库连接里进行。

猜你喜欢

转载自blog.csdn.net/qq_37520037/article/details/83217210
今日推荐