Java中Object.wait()和LockSupport.park()的用法
Object.wait()
和 LockSupport.park()
都是用来使当前线程等待的方法,但它们在使用场景和机制上有所不同:
Object.wait()
用途:
wait()
方法属于对象监视器(Monitor)的一部分,通常与synchronized
块或方法一起使用。当线程调用某个对象的wait()
方法时,它会释放该对象的锁,并使自己进入等待状态,直到其他线程调用该对象的notify()
或notifyAll()
方法唤醒它,此时线程会重新尝试获取锁并继续执行。位置限制:只能在同步代码块或同步方法中调用,因为需要先获取到对象的监视器锁。
唤醒条件:可以被
notify()
唤醒,意味着有一个或多个等待线程会被唤醒,但具体哪个线程被唤醒是不确定的;也可以被notifyAll()
唤醒,这时所有等待该对象监视器的线程都会进入锁的竞争状态。
LockSupport.park()
用途:
park()
方法属于java.util.concurrent.locks.LockSupport
类,它提供了一种低级别的线程阻塞原语。它不需要与特定的锁关联,可以在任何地方调用,使得线程阻塞。它通常与unpark()
方法配对使用,后者可以唤醒一个调用了park()
的线程。位置限制:没有位置限制,可以在任何地方调用,不需要先获取锁。
唤醒条件:调用
LockSupport.unpark(Thread thread)
方法可以直接唤醒目标线程,更加灵活和精确。它可以唤醒一个特定的线程,而无需竞争或不确定性。线程许可:
park()
和unpark()
是基于每个线程的许可(permit)机制。初始时,每个线程没有许可,调用unpark()
会给指定线程添加一个许可,即使之前已经调用过unpark()
给该线程添加了许可,再调用也不会造成影响(许可不会累积)。调用park()
时,如果没有许可,线程会阻塞,如果有许可,则消耗许可并继续执行。
总结来说,Object.wait()
更适合与同步代码块或方法一起使用,与对象的监视器锁紧密相关,适用于传统的线程协作场景;而 LockSupport.park()
提供了一种更底层、更灵活的线程阻塞和唤醒机制,适用于更复杂的并发控制逻辑。
下面是分别使用 Object.wait()
和 LockSupport.park()
的简单示例:
使用 Object.wait() 的例子
public class WaitNotifyExample { public static void main(String[] args) { final Object monitor = new Object(); Thread waitingThread = new Thread(new Runnable() { @Override public void run() { synchronized (monitor) { System.out.println(Thread.currentThread().getName() + " 开始等待"); try { monitor.wait(); // 等待被唤醒 } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName() + " 被唤醒"); } } }, "WaitingThread"); Thread notifierThread = new Thread(new Runnable() { @Override public void run() { try { Thread.sleep(2000); // 模拟一些工作 } catch (InterruptedException e) { e.printStackTrace(); } synchronized (monitor) { monitor.notify(); // 唤醒等待的线程 System.out.println(Thread.currentThread().getName() + " 发出了唤醒通知"); } } }, "NotifierThread"); waitingThread.start(); notifierThread.start(); } }
使用 LockSupport.park() 的例子
import java.util.concurrent.locks.LockSupport; public class ParkUnparkExample { public static void main(String[] args) { Thread waitingThread = new Thread(new Runnable() { @Override public void run() { System.out.println(Thread.currentThread().getName() + " 开始等待"); LockSupport.park(); // 阻塞当前线程 System.out.println(Thread.currentThread().getName() + " 被唤醒"); } }, "WaitingThread"); Thread unparkerThread = new Thread(new Runnable() { @Override public void run() { try { Thread.sleep(2000); // 模拟一些工作 } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName() + " 准备唤醒线程"); LockSupport.unpark(waitingThread); // 唤醒指定线程 } }, "UnparkerThread"); waitingThread.start(); unparkerThread.start(); } }
在这两个例子中,第一个展示了如何使用 Object.wait()
和 notify()
来实现线程间的等待与通知,第二个例子展示了如何使用 LockSupport.park()
和 unpark()
实现类似的线程控制功能,但更为灵活和独立于特定锁。
到此这篇关于Java中Object.wait()和LockSupport.park()的用法的文章就介绍到这了,更多相关Java Object.wait() LockSupport.park()内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
Java Swing CardLayout卡片布局的实现示例
这篇文章主要介绍了Java Swing CardLayout卡片布局的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧2019-12-12java字符串日期类Date和Calendar相互转化及相关常用方法
Java语言的Calendar(日历),Date(日期),和DateFormat(日期格式)组成了Java标准的一个基本但是非常重要的部分,下面这篇文章主要给大家介绍了关于java字符串日期类Date和Calendar相互转化及相关常用方法的相关资料,需要的朋友可以参考下2023-12-12Mybatis报错mapkey is required问题及解决
这篇文章主要介绍了Mybatis报错mapkey is required问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教2024-06-06
最新评论