在Java中使用线程的时候肯定会有线程挂起的这种情况出现,在Java中提供了3种方式:suspend/resume、wait/notify,notifyAll、park/unpark。
1.suspend/resume
这种方式已经在Java中被弃用,因为它容易引起死锁。在使用关键字synchronized的时候如
synchronized (this) { Thread.currentThread().suspend(); }
这个时候使用resume方法是无法唤醒线程的,还有一种情况是如果在resume方法之后才进行suspend方法的调用,这个也是无法进行suspend方法的唤醒的,因为这个事件发生的顺序已经造成了死锁的出现。
2.wait/notify,notifyAll
这个是Java中常用的线程挂起方法,当调用wait方法的时候线程会自动的释放掉占有的线程资源锁,然后通过notify或notifyAll方法进行wait方法的唤醒,因此在这个地方不会出现死锁,当时如suspend/resume中提到的,如果在notify或notifyAll方法之后在进行wait操作,那么肯定也是会出现死锁的。这里要说明一点的是wait方式的线程挂起必须和synchronized关键字一起使用才可以,不然会出现java.lang.IllegalMonitorStateException异常,并且synchronized关键字的锁是相同的锁才能进行解锁操作,如下
InterruptTest test = new InterruptTest(); new Thread(() ->{ synchronized (test){ try { System.out.println(1); test.wait(); System.out.println(2); } catch (InterruptedException e) { e.printStackTrace(); } } }).start(); new Thread(() ->{ LockSupport.parkNanos(1000*1000*1000*2); synchronized (test){ try { System.out.println(3); test.notify(); Thread.sleep(1000); System.out.println(4); } catch (InterruptedException e) { e.printStackTrace(); } } }).start();
这里必须说明一下的是sleep方法,虽然它也能将线程挂起,但是它会产生InterruptedException异常,当sleep一定时间后它将会自动执行后面的方法,也可以通过interrupt方法进行主动打断sleep方法进行线程唤醒。同时需要注意的是wait()方法会破坏原子性,因为当前线程挂起后,其他线程就可以修改wait()方法后的变量,因此就会出现原子性数据的破坏。
3.park/unpark
park的字面量意思是指停车场的意思,使用park来挂起线程后需要调用unpark来进行唤醒,这个没有先后顺序的区分,如果你提前进行了unpark,然后在进行park也是可以的,但是提前了的多个unpark只能看做是一个unpark,不能进行重复叠加,如果再次park的话需要新的unpark来进行唤醒操作,这个比如你在停车场进行停车操作,你如果提前进行了预约停车,你在未进入停车场之前都是可以进行多次预约的,这所有的预约只扣一次钱(unpark),但是这所有的预约操作都看作是你这一次进入停车场停车(park)的凭证,如果你离开了通过缴费凭证(unpark)一旦你想进行下一次停车,那么已经使用过的预约都不能进行作数了,你只能再次预约或者直接进入停车场,一旦要离开只能再次缴费(unpark)。
调用park/unpark使用的是LockSupport.park()/LockSupport.unpark()。
使用park/unpark的使用使用synchronized关键字也会出现死锁的情况,因为它并不释放线程所占用的锁资源,所以使用的时候也需要注意。
相关推荐
Java中线程挂起的几种方式详解 Java中线程挂起的几种方式详解是Java编程中一个常见的技术,主要用于控制线程的执行和暂停。在Java中提供了三种方式来实现线程挂起:suspend/resume、wait/notify、notifyAll和park/...
Java 多线程的几种实现方法总结 Java 多线程是指在同一个程序中可以有多条执行线索同时执行,每个线程上都关联有要执行的代码,即可以有多段程序代码同时运行。为了帮助大家更好地理解和掌握 Java 多线程的知识,...
Java线程具有以下几种基本状态: - **新建状态(New)**:当创建了一个Thread类的子类实例时,线程处于新建状态。 - **就绪状态(Runnable)**:线程启动后,会进入就绪状态,等待CPU分配时间片。 - **挂起状态...
在 Java 中通过 wait(),notify(),notifyAll() 来实现,这三个方法是在 Object 类中定义的,当你想让线程挂起的时候调用 obj.wait() 方法,在同样的 obj 上调用 notify() 则让线程重新开始运行。 最后,以 SUN ...
4. 条件变量(Condition Variable)模式:这是一种高级的线程间同步机制,允许线程在等待某个条件成立时挂起,条件成立时由其他线程通知。 四、并发集合框架 Java提供了丰富的并发集合类,以支持多线程环境下对集合...
在Java中,线程间通信(IPC)是通过特定的机制实现的,主要包括以下几种方式: a) 管道(Pipes):Java提供了java.io.PipedInputStream和java.io.PipedOutputStream类来实现线程间的字节流通信。 b) 原子变量...
Java NIO(New IO)是Java 1.4版本引入的一个新特性,它为Java提供了一种不同于传统IO( Blocking I/O)的处理I/O操作的方式。传统的IO模型基于流和缓冲区,采用同步阻塞的方式,即在读写数据时会阻塞当前线程,直到...
二、创建线程的几种方式 1. 继承Thread类:创建一个新的类,继承自Thread类,并重写其run()方法。实例化后调用start()启动线程。 ```java class MyThread extends Thread { public void run() { // 代码执行体 } ...
- 阻塞(Blocked):线程因I/O操作、锁等待等原因被挂起,等待恢复运行。 - 终止(Terminated):`run()`方法执行完毕或者线程被显式地中断(`interrupt()`)。 5. **线程同步与通信**: - 为了避免多线程环境下...
一个线程可以被启动、运行、挂起、恢复和终止。 2. **创建线程**:有多种方法可以创建线程,最常见的两种方式是继承Thread类和实现Runnable接口。此外,还可以通过Callable接口与Executor框架来实现线程。 3. **...
2. **避免阻塞问题**:例如,在读取用户输入时使用`nextLine()`等阻塞方法,如果这些操作发生在主线程中,则会导致整个程序挂起等待用户输入。通过将这些操作放在单独的线程中,可以确保主线程或其他线程继续执行不...
`suspend`方法将线程挂起,但不释放其持有的任何锁,可能导致死锁情况发生。`resume`方法用于恢复由`suspend`方法挂起的线程。现代的线程管理推荐使用更安全的方法,如`interrupt`机制。 #### interrupt,...
5. 挂起状态:使用`suspend()`方法使线程暂停,释放资源,进入挂起队列,等待`resume()`方法恢复。 6. 死亡状态:`run()`方法执行完毕或`stop()`方法被调用后,线程结束。 线程的调度策略包括优先级调度和时间片...
在Java中,创建线程主要有以下几种方式: 1. 实现Runnable接口:创建一个新的类,实现Runnable接口,并重写run()方法。然后,将这个类的实例作为参数传递给Thread类的构造函数,创建线程对象并调用start()方法启动。...
在Java中,创建线程主要有以下几种方式: 1. **继承Thread类**: - 定义一个类继承自`Thread`类,并重写`run()`方法。 - 创建该类的对象并调用其`start()`方法启动线程。 2. **实现Runnable接口**: - 定义一...
- `Thread.suspend()`(已废弃):不再推荐使用,它会将线程挂起,但不释放任何锁,可能导致死锁。 - `object.wait()`:当前线程放弃对象锁,进入等待队列,直到其他线程调用`notify()`或`notifyAll()`唤醒。必须...
Java线程错误捕获工具CheckThread是一款专为Java开发者设计的实用工具,它主要用于帮助开发者在多线程环境中更有效地捕获和管理错误。在Java编程中,多线程是常见的并发执行方式,但同时也带来了错误处理的复杂性。...
Quasar 是一个为 Java 和 Kotlin 设计的高性能轻量级纤程库,它提供了一种类似于 Go 语言中的 channel 以及 Erlang 的 actor 模型的编程方式,并且支持其他的异步编程特性。通过 Quasar,开发人员可以在 Java 和 ...