【Java】通俗易懂的JUC源码剖析-LockSupport

前言

LockSupport是rt.jar下的工具类,它的作用是挂起和唤醒线程,它在JUC很多同步组件中都会用到,比如AQS。LockSupport类与每个使用它的线程都会关联一个许可证,在默认情况下调用LockSupport类方法的线程是没有许可证的。LockSupport内部是用Unsafe的park和unpark方法实现的。

主要方法

(1)void park()

调用park()方法会阻塞当前线程,除非当前线程已经拿到了与LockSupport关联的许可证。当其他线程调用unpark(Thread thread)方法并且将当前线程作为参数时,或者调用当前线程的interrupt()方法时,当前线程会从阻塞状态返回。但是,需要注意的是,与调用sleep(),wait()等方法不同,调用park()方法被中断返回时,不会抛出InterruptedException异常。还有需要注意的是,park()方法返回时不会告诉你是什么原因返回(unpark或interrupt),但我们可以根据调用前后的中断标志是否改变来判断是否因为interrupt返回的。

(2)void unpark(Thread thread)

线程调用unpark方法时,如果参数线程thread没有持有许可证,则让其持有,如果thread在此前因调用park而被阻塞,则调用unpark后,该线程会被唤醒。如果thread之前没有调用park,则调用unpark后,再调用park,线程会立即返回,而不会阻塞挂起。用代码来说明:

import java.util.concurrent.locks.LockSupport;

public class UnparkDemo {

public static void main(String[] args) {

System.out.println("give current thread a permit");

LockSupport.unpark(Thread.currentThread());

System.out.println("current thread begin park");

LockSupport.park();

System.out.println("current thread end park");

}

}

输出结果如下:

当前线程调用park方法后并没有阻塞挂起,而是立即返回了,因为此前调用unpark给了它许可证。

【Java】通俗易懂的JUC源码剖析-LockSupport

(3)void parkNanos(long nanos)

和park方法类似,但有个超时参数,如果等待了nanos(单位纳秒)时间后,当前线程没有被唤醒,会自动返回。

(4)void park(Object blocker)

当前线程调用park(blocker)被挂起时,blocker对象会被记录在当前线程内部,它的作用是当我们使用诊断工具(如jstack等)dump线程时,可以观察到线程被阻塞在哪个类上面。诊断工具是通过getBlocker()拿到blocker对象的。因此JDK推荐我们使用带有blocker参数的park方法,例如LockSupport.park(this)。同样用代码来说明下:

import java.util.concurrent.locks.LockSupport;

public class ParkWithBlockerDemo {

public void parkWithBlocker() {

LockSupport.park(this);

}

public static void main(String[] args) {

ParkWithBlockerDemo demo = new ParkWithBlockerDemo();

demo.parkWithBlocker();

}

}

程序运行后,线程会被阻塞挂起,此时先用jps命令找到它的pid,然后用jstack pid执行,会有如下结果:

【Java】通俗易懂的JUC源码剖析-LockSupport

不加blocker参数的话,是不会有这个标记的,感兴趣的同学可以去试试。

park(blocker)方法源码:

public static void park(Object blocker) {

Thread t = Thread.currentThread();

setBlocker(t, blocker);

UNSAFE.park(false, 0L);

setBlocker(t, null);

}

代码逻辑很清晰明了,原理就是真正park之前,将blocker对象绑定在当前Thread内部的某个变量中,等park返回后,解除绑定。其中setBlocker源码如下:

private static void setBlocker(Thread t, Object arg) {

// Even though volatile, hotspot doesn't need a write barrier here.

UNSAFE.putObject(t, parkBlockerOffset, arg);

}

其中parkBlockerOffset是Thread类parkBlocker变量的内存偏移量。

private static final long parkBlockerOffset;

parkBlockerOffset = UNSAFE.objectFieldOffset

(tk.getDeclaredField("parkBlocker"));

(5)parkUntil(long deadline)

与parkNanos不同的是,deadline是个绝对时间戳,表示到某个时间点后,当前线程会被唤醒。

结束语

最后再来看个栗子来加深LockSupport的理解。

import java.util.Queue;

import java.util.concurrent.ConcurrentLinkedDeque;

import java.util.concurrent.atomic.AtomicBoolean;

import java.util.concurrent.locks.LockSupport;

public class FIFOMutex {

private final AtomicBoolean locked = new AtomicBoolean(false);

private final Queue<Thread> waiters = new ConcurrentLinkedDeque<>();

public void lock() {

boolean wasInterrupted = false;

Thread current = Thread.currentThread();

// 把当前线程放到队列末尾

waiters.add(current);

// 只有队列头部的线程才能获取锁

while (waiters.peek() != current || locked.compareAndSet(false, true)) {

LockSupport.park(this);

// 记录是否被中断过,并重置中断标志,这里其实就是忽略中断

if (Thread.interrupted()) {

wasInterrupted = true;

}

}

// 移除队列头部元素,即成功获取锁的当前线程

waiters.remove();

// 重新响应之前的中断

if (wasInterrupted) {

current.interrupt();

}

}

public void unlock() {

locked.set(false);

// 唤醒原本处于队列第二,现在处于头部的线程,让它尝试获取锁

LockSupport.unpark(waiters.peek());

}

}

这是一个先入先出的锁,也就是只有队列的头部线程才能获取锁,如果当前线程不是头部元素,或者锁已经被其他线程获取,则调用park方法阻塞自己。

这里的wasInterrupted怎么理解呢?如果park方法是因为被中断而返回的,则忽略中断,只记录wasInterrupted=true。因为这说明不是其他线程调用了解锁方法unlock()里面的unpark让它返回的,它还不能获取锁,得继续阻塞。虽然当前线程对中断标志不感兴趣,但不代表其他线程也不敢兴趣,所以在成功获取锁后,重新对自己调用interrupt,恢复下中断标志的值。

以上是 【Java】通俗易懂的JUC源码剖析-LockSupport 的全部内容, 来源链接: utcz.com/a/110358.html

回到顶部