Java并发编程——锁(公平锁,非公平锁、可重入锁、自旋锁、死锁)

一、公平锁,非公平锁

公平锁: 非常公平, 不能够插队,必须先来后到!
非公平锁:非常不公平,可以插队 (默认都是非公平)

public ReentrantLock() {
	sync = new NonfairSync();
}
public ReentrantLock(boolean fair) {
	sync = fair ? new FairSync() : new NonfairSync();
}

二、可重入锁

可重入就是说某个线程已经获得某个锁,可以再次获取锁而不会出现死锁
Synchronized:

// Synchronized
public class Demo01 {
	public static void main(String[] args) {
		Phone phone = new Phone();
		new Thread(()->{
			phone.sms();
		},"A").start();
		new Thread(()->{
			phone.sms();
		},"B").start();
	}
}
class Phone{
	public synchronized void sms(){
		System.out.println(Thread.currentThread().getName() + "sms");
		call(); // 这里也有锁
	}
	public synchronized void call(){
		System.out.println(Thread.currentThread().getName() + "call");
	}
}

lock:

public class Demo02 {
	public static void main(String[] args) {
		Phone2 phone = new Phone2();
		new Thread(()->{
			phone.sms();
		},"A").start();
		new Thread(()->{
			phone.sms();
		},"B").start();
	}
}
class Phone2{
	Lock lock = new ReentrantLock();
	public void sms(){
		lock.lock(); // 细节问题:lock.lock(); lock.unlock(); // lock 锁必须配对,否则就会死在里面
		lock.lock();
		try {
			System.out.println(Thread.currentThread().getName() + "sms");
			call(); // 这里也有锁
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			lock.unlock();
			lock.unlock();
		}
	}
	public void call(){
		lock.lock();
		try {
			System.out.println(Thread.currentThread().getName() + "call");
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			lock.unlock();
		}
	}
}

三、自旋锁

自旋锁:是指当一个线程在获取锁的时候,如果锁已经被其它线程获取,那么该线程将循环等待,然后不断的判断锁是否能够被成功获取,直到获取到锁才会退出循环。

自定义lock()和unlock()

/**
* 自旋锁
*/
public class SpinlockDemo {
	// int 0
	// Thread null
	AtomicReference<Thread> atomicReference = new AtomicReference<>();
	// 加锁
	public void myLock(){
		Thread thread = Thread.currentThread();
		System.out.println(Thread.currentThread().getName() + "--> mylock");
	// 自旋锁
		while (!atomicReference.compareAndSet(null,thread)){
	}
}
	// 解锁
	// 加锁
	public void myUnLock(){
		Thread thread = Thread.currentThread();
		System.out.println(Thread.currentThread().getName() + "--> myUnlock");
		atomicReference.compareAndSet(thread,null);
	}
}

测试:

public class TestSpinLock {
	public static void main(String[] args) throws InterruptedException {
	// 底层使用的自旋锁CAS
		SpinlockDemo lock = new SpinlockDemo();
		new Thread(()-> {
			lock.myLock();
			try {
				TimeUnit.SECONDS.sleep(5);
			} catch (Exception e) {
				e.printStackTrace();
			} finally {
				lock.myUnLock();
			}
		},"T1").start();
		TimeUnit.SECONDS.sleep(1);
		new Thread(()-> {
			lock.myLock();
			try {
				TimeUnit.SECONDS.sleep(1);
			} catch (Exception e) {
				e.printStackTrace();
			} finally {
				lock.myUnLock();
			}
		},"T2").start();
	}
}

在这里插入图片描述

四、死锁

关于死锁的具体内容查看下面的链接
https://blog.youkuaiyun.com/qq_43576814/article/details/119685810

死锁排查:
1.打开cmd,输入jps -l
在这里插入图片描述
2.输入jstack 进程号
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Alchemy_Ding

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值