我们在写并发程序的时候,一个非常常见的需求就是保证在某一个时刻只有一个线程执行某段代码,像这种代码叫做临界区,而通常保证一个时刻只有一个线程执行临界区的代码的方法就是锁🔒。在本篇文章当中我们将会仔细分析和学习自旋锁,所谓自旋锁就是通过while循环实现的,让拿到锁的线程进入临界区执行代码,让没有拿到锁的线程一直进行while死循环,这其实就是线程自己“旋”在while循环了,因而这种锁就叫做自旋锁。
在谈自旋锁之前就不得不谈原子性了。所谓原子性简单说来就是一个一个操作要么不做要么全做,全做的意思就是在操作的过程当中不能够被中断,比如说对变量data
进行加一操作,有以下三个步骤:
data
从内存加载到寄存器。data
这个值加一。原子性就表示一个线程在进行加一操作的时候,不能够被其他线程中断,只有这个线程执行完这三个过程的时候其他线程才能够操作数据data
。
我们现在用代码体验一下,在Java当中我们可以使用AtomicInteger
进行对整型数据的原子操作:
import java.util.concurrent.atomic.AtomicInteger;
public class AtomicDemo {
public static void main(String[] args) throws InterruptedException {
AtomicInteger data = new AtomicInteger();
data.set(0); // 将数据初始化位0
Thread t1 = new Thread(() -> {
for (int i = 0; i < 100000; i++) {
data.addAndGet(1); // 对数据 data 进行原子加1操作
}
});
Thread t2 = new Thread(() -> {
for (int i = 0; i < 100000; i++) {
data.addAndGet(1);// 对数据 data 进行原子加1操作
}
});
// 启动两个线程
t1.start();
t2.start();
// 等待两个线程执行完成
t1.join();
t2.join();
// 打印最终的结果
System.out.println(data); // 200000
}
}
从上面的代码分析可以知道,如果是一般的整型变量如果两个线程同时进行操作的时候,最终的结果是会小于200000。
我们现在来模拟一下一般的整型变量出现问题的过程:
主内存data
的初始值等于0,两个线程得到的data
初始值都等于0。
现在线程一将data
加一,然后线程一将data
的值同步回主内存,整个内存的数据变化如下:
现在线程二data
加一,然后将data
的值同步回主内存(将原来主内存的值覆盖掉了):
我们本来希望data
的值在经过上面的变化之后变成2
,但是线程二覆盖了我们的值,因此在多线程情况下,会使得我们最终的结果变小。
但是在上面的程序当中我们最终的输出结果是等于20000的,这是因为给data
进行+1
的操作是原子的不可分的,在操作的过程当中其他线程是不能对data
进行操作的。这就是原子性带来的优势。
现在我们已经了解了原子性的作用了,我们现在来了解AtomicInteger
类的另外一个原子性的操作——compareAndSet
,这个操作叫做比较并交换(CAS),他具有原子性。
public static void main(String[] args) {
AtomicInteger atomicInteger = new AtomicInteger();
atomicInteger.set(0);
atomicInteger.compareAndSet(0, 1);
}
compareAndSet函数的意义:首先会比较第一个参数(对应上面的代码就是0)和atomicInteger的值,如果相等则进行交换,也就是将atomicInteger的值设置为第二个参数(对应上面的代码就是1),如果这些操作成功,那么compareAndSet函数就返回true
,如果操作失败则返回false
,操作失败可能是因为第一个参数的值(期望值)和atomicInteger不相等,如果相等也可能因为在更改atomicInteger的值的时候失败(因为可能有多个线程在操作,因为原子性的存在,只能有一个线程操作成功)。
我们可以使用AtomicInteger类实现自旋锁,我们可以用0这个值表示未上锁,1这个值表示已经上锁了。
atomicInteger.compareAndSet(0, 1)
进行实现,我们在前面已经提到了只能够有一个线程完成这个操作,也就是说只能有一个线程调用这行代码然后返回true
其余线程都返回false
,这些返回false
的线程不能够进入临界区,因此我们需要这些线程停在atomicInteger.compareAndSet(0, 1)
这行代码不能够往下执行,我们可以使用while循环让这些线程一直停在这里while (!value.compareAndSet(0, 1));
,只有返回true
的线程才能够跳出循环,其余线程都会一直在这里循环,我们称这种行为叫做自旋,这种锁因而也被叫做自旋锁。value.compareAndSet(1, 0);
就可以实现,将锁的状态还原为未上锁的状态,这样其他的自旋的线程就可以拿到锁,然后进入临界区了。import java.util.concurrent.atomic.AtomicInteger;
public class SpinLock {
// 0 表示未上锁状态
// 1 表示上锁状态
protected AtomicInteger value;
public SpinLock() {
this.value = new AtomicInteger();
// 设置 value 的初始值为0 表示未上锁的状态
this.value.set(0);
}
public void lock() {
// 进行自旋操作
while (!value.compareAndSet(0, 1));
}
public void unlock() {
// 将锁的状态设置为未上锁状态
value.compareAndSet(1, 0);
}
}
上面就是我们自己实现的自旋锁的代码,这看起来实在太简单了,但是它确实帮助我们实现了一个锁,而且能够在真实场景进行使用的,我们现在用代码对上面我们写的锁进行测试。
测试程序:
public class SpinLockTest {
public static int data;
public static SpinLock lock = new SpinLock();
public static void add() {
for (int i = 0; i < 100000; i++) {
// 上锁 只能有一个线程执行 data++ 操作 其余线程都只能进行while循环
lock.lock();
data++;
lock.unlock();
}
}
public static void main(String[] args) throws InterruptedException {
Thread[] threads = new Thread[100];
// 设置100个线程
for (int i = 0; i < 100; i ++) {
threads[i] = new Thread(SpinLockTest::add);
}
// 启动一百个线程
for (int i = 0; i < 100; i++) {
threads[i].start();
}
// 等待这100个线程执行完成
for (int i = 0; i < 100; i++) {
threads[i].join();
}
System.out.println(data); // 10000000
}
}
在上面的代码单中,我们使用100个线程,然后每个线程循环执行100000data++
操作,上面的代码最后输出的结果是10000000,和我们期待的结果是相等的,这就说明我们实现的自旋锁是正确的。
在上面实现的自旋锁当中已经可以满足一些我们的基本需求了,就是一个时刻只能够有一个线程执行临界区的代码。但是上面的的代码并不能够满足重入的需求,也就是说上面写的自旋锁并不是一个可重入的自旋锁,事实上在上面实现的自旋锁当中重入的话就会产生死锁。
我们通过一份代码来模拟上面重入产生死锁的情况:
public static void add(int state) throws InterruptedException {
TimeUnit.SECONDS.sleep(1);
if (state <= 3) {
lock.lock();
System.out.println(Thread.currentThread().getName() + "\t进入临界区 state = " + state);
for (int i = 0; i < 10; i++)
data++;
add(state + 1); // 进行递归重入 重入之前锁状态已经是1了 因为这个线程进入了临界区
lock.unlock();
}
}
state
的值为1,那么在线程执行for循环之后再次递归调用add
函数的话,那么state
的值就变成了2。针对上面这种情况我们需要实现一个可重入的自旋锁,我们的思想大致如下:
owner
一个用于存当前拥有锁的线程,count
一个记录当前线程进入锁的次数。owner = Thread.currentThread()
并且count = 1
。owner
是不是指向自己,则一直进行循环操作,如果是则直接进行count++
操作,然后就可以进入临界区了。count
大于一的话,说明这个线程重入了这把锁,因此不能够直接将锁设置为0也就是未上锁的状态,这种情况直接进行count--
操作,如果count
等于1的话,说明线程当前的状态不是重入状态(可能是重入之后递归返回了),因此在出临界区之前需要将锁的状态设置为0,也就是没上锁的状态,好让其他线程能够获取锁。实现的可重入锁代码如下:
public class ReentrantSpinLock extends SpinLock {
private Thread owner;
private int count;
@Override
public void lock() {
if (owner == null || owner != Thread.currentThread()) {
while (!value.compareAndSet(0, 1));
owner = Thread.currentThread();
count = 1;
}else {
count++;
}
}
@Override
public void unlock() {
if (count == 1) {
count = 0;
value.compareAndSet(1, 0);
}else
count--;
}
}
下面我们通过一个递归程序去验证我们写的可重入的自旋锁是否能够成功工作。
测试程序:
import java.util.concurrent.TimeUnit;
public class ReentrantSpinLockTest {
public static int data;
public static ReentrantSpinLock lock = new ReentrantSpinLock();
public static void add(int state) throws InterruptedException {
TimeUnit.SECONDS.sleep(1);
if (state <= 3) {
lock.lock();
System.out.println(Thread.currentThread().getName() + "\t进入临界区 state = " + state);
for (int i = 0; i < 10; i++)
data++;
add(state + 1);
lock.unlock();
}
}
public static void main(String[] args) throws InterruptedException {
Thread[] threads = new Thread[10];
for (int i = 0; i < 10; i++) {
threads[i] = new Thread(new Thread(() -> {
try {
ReentrantSpinLockTest.add(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
}, String.valueOf(i)));
}
for (int i = 0; i < 10; i++) {
threads[i].start();
}
for (int i = 0; i < 10; i++) {
threads[i].join();
}
System.out.println(data);
}
}
上面程序的输出:
Thread-3 进入临界区 state = 1
Thread-3 进入临界区 state = 2
Thread-3 进入临界区 state = 3
Thread-0 进入临界区 state = 1
Thread-0 进入临界区 state = 2
Thread-0 进入临界区 state = 3
Thread-9 进入临界区 state = 1
Thread-9 进入临界区 state = 2
Thread-9 进入临界区 state = 3
Thread-4 进入临界区 state = 1
Thread-4 进入临界区 state = 2
Thread-4 进入临界区 state = 3
Thread-7 进入临界区 state = 1
Thread-7 进入临界区 state = 2
Thread-7 进入临界区 state = 3
Thread-8 进入临界区 state = 1
Thread-8 进入临界区 state = 2
Thread-8 进入临界区 state = 3
Thread-5 进入临界区 state = 1
Thread-5 进入临界区 state = 2
Thread-5 进入临界区 state = 3
Thread-2 进入临界区 state = 1
Thread-2 进入临界区 state = 2
Thread-2 进入临界区 state = 3
Thread-6 进入临界区 state = 1
Thread-6 进入临界区 state = 2
Thread-6 进入临界区 state = 3
Thread-1 进入临界区 state = 1
Thread-1 进入临界区 state = 2
Thread-1 进入临界区 state = 3
300
从上面的输出结果我们就可以知道,当一个线程能够获取锁的时候他能够进行重入,而且最终输出的结果也是正确的,因此验证了我们写了可重入自旋锁是有效的!
在本篇文章当中主要给大家介绍了自旋锁和可重入自旋锁的原理,并且实现了一遍,其实代码还是比较简单关键需要大家将这其中的逻辑理清楚:
AtomicInteger
类,并且我们使用0和1这两个数值用于表示无锁和锁被占用两个状态,在获取锁的时候使用while循环不断进行CAS操作,直到操作成功返回true
,在释放锁的时候使用CAS将锁的状态从1变成0。count = 1
的情况才能将锁的状态从1设置成0。以上就是本篇文章的所有内容了,我是LeHung,我们下期再见!!!更多精彩内容合集可访问项目:https://github.com/Chang-LeHung/CSCore
关注公众号:一无是处的研究僧,了解更多计算机(Java、Python、计算机系统基础、算法与数据结构)知识。