程序是很简易的。然而,在编程人员面前,多线程呈现出了一组新的难题,如果没有被恰当的解决,将导致意外的行为以及细微的、难以发现的错误。
在本篇文章中,我们针对这些难题之一:如何中断一个正在运行的线程。
背景中断(Interrupt)一个线程意味着在该线程完成任务之前停止其正在进行的一切,有效地中止其当前的操作。线程是死亡、还是等待新的任务或是 继续运行至下一步,就取决于这个程序。虽然初次看来它可能显得简单,但是,你必须进行一些预警以实现期望的结果。你最好还是牢记以下的几点告诫。
首先,忘掉Thread.stop方法。虽然它确实停止了一个正在运行的线程,然而,这种方法是不安全也是不受提倡的,这意味着,在未来的JAVA版本中,它将不复存在。
一些轻率的家伙可能被另一种方法Thread.interrupt所迷惑。尽管,其名称似乎在暗示着什么,然而,这种方法并不会中断一个正在运行的线程 (待会将进一步说明),正如Listing A中描述的那样。它创建了一个线程,并且试图使用Thread.interrupt方法停止该线程。 Thread.sleep()方法的调用,为线程的初始化和中止提供了充裕的时间。线程本身并不参与任何有用的操作。
- class Example1 extends Thread {
- boolean stop=false;
- public static void main( String args[] ) throws Exception {
- Example1 thread = new Example1();
- System.out.println( "Starting thread..." );
- thread.start();
- Thread.sleep( 3000 );
- System.out.println( "Interrupting thread..." );
- thread.interrupt();
- Thread.sleep( 3000 );
- System.out.println("Stopping application..." );
- //System.exit(0);
- }
- public void run() {
- while(!stop){
- System.out.println( "Thread is running..." );
- long time = System.currentTimeMillis();
- while((System.currentTimeMillis()-time < 1000)) {
- }
- }
- System.out.println("Thread exiting under request..." );
- }
- }
如果你运行了Listing A中的代码,你将在控制台看到以下输出:
- Starting thread...
- Thread is running...
- Thread is running...
- Thread is running...
- Interrupting thread...
- Thread is running...
- Thread is running...
- Thread is running...
- Stopping application...
- Thread is running...
- Thread is running...
- Thread is running...
- ...............................
甚至,在Thread.interrupt()被调用后,线程仍然继续运行。
真正地中断一个线程
中断线程最好的,最受推荐的方式是,使用共享变量(shared variable)发出信号,告诉线程必须停止正在运行的任务。线程必须周期性的核查这一变量(尤其在冗余操作期间),然后有秩序地中止任务。Listing B描述了这一方式。
- /*Listing B*/
- class Example2 extends Thread {
- volatile boolean stop = false;
- public static void main( String args[] ) throws Exception {
- Example2 thread = new Example2();
- System.out.println( "Starting thread..." );
- thread.start();
- Thread.sleep( 3000 );
- System.out.println( "Asking thread to stop..." );
- thread.stop = true;
- Thread.sleep( 3000 );
- System.out.println( "Stopping application..." );
- //System.exit( 0 );
- }
- public void run() {
- while ( !stop ) {
- System.out.println( "Thread is running..." );
- long time = System.currentTimeMillis();
- while ( (System.currentTimeMillis()-time < 1000) && (!stop) ) {
- }
- }
- System.out.println( "Thread exiting under request..." );
- }
- }
运行Listing B中的代码将产生如下输出(注意线程是如何有秩序的退出的)
- Starting thread...
- Thread is running...
- Thread is running...
- Thread is running...
- Asking thread to stop...
- Thread exiting under request...
- Stopping application...
虽然该方法要求一些编码,但并不难实现。同时,它给予线程机会进行必要的清理工作,这在任何一个多线程应用程序中都是绝对需要的。请确认将共享变 量定义成volatile 类型或将对它的一切访问封入同步的块/方法(synchronized blocks/methods)中。
到目前为止一切顺利!但是,当线程等待某些事件发生而被阻塞,又会发生什么?当然,如果线程被阻塞,它便不能核查共享变量,也就不能停止。这在许多情况 下会发生,例如调用Object.wait()、ServerSocket.accept()和DatagramSocket.receive()时,这 里仅举出一些。
他们都可能永久的阻塞线程。即使发生超时,在超时期满之前持续等待也是不可行和不适当的,所以,要使用某种机制使得线程更早地退出被阻塞的状态。
很不幸运,不存在这样一种机制对所有的情况都适用,但是,根据情况不同却可以使用特定的技术。在下面的环节,我将解答一下最普遍的例子。
使用Thread.interrupt()中断线程
正如Listing A中所描述的,Thread.interrupt()方法不会中断一个正在运行的线程。这一方法实际上完成的是,在线程受到阻塞时抛出一个中断信号,这样 线程就得以退出阻塞的状态。更确切的说,如果线程被Object.wait, Thread.join和 Thread.sleep三种方法之一阻塞,那么,它将接收到一个中断异常(InterruptedException),从而提早地终结被阻塞状态。
因此,如果线程被上述几种方法阻塞,正确的停止线程方式是设置共享变量,并调用interrupt()(注意变量应该先设置)。如果线程没有被阻塞, 这时调用interrupt()将不起作用;否则,线程就将得到异常(该线程必须事先预备好处理此状况),接着逃离阻塞状态。在任何一种情况中,最后线程 都将检查共享变量然后再停止。Listing C这个示例描述了该技术。
- /*Listing C*/
- class Example3 extends Thread {
- volatile boolean stop = false;
- public static void main( String args[] ) throws Exception {
- Example3 thread = new Example3();
- System.out.println( "Starting thread..." );
- thread.start();
- Thread.sleep( 3000 );
- System.out.println( "Asking thread to stop..." );
- thread.stop = true;//如果线程阻塞,将不会检查此变量
- thread.interrupt();
- Thread.sleep( 3000 );
- System.out.println( "Stopping application..." );
- //System.exit( 0 );
- }
- public void run() {
- while ( !stop ) {
- System.out.println( "Thread running..." );
- try {
- Thread.sleep( 1000 );
- } catch ( InterruptedException e ) {
- System.out.println( "Thread interrupted..." );
- }
- }
- System.out.println( "Thread exiting under request..." );
- }
- }
一旦Listing C中的Thread.interrupt()被调用,线程便收到一个异常,于是逃离了阻塞状态并确定应该停止。运行以上代码将得到下面的输出:
- Starting thread...
- Thread running...
- Thread running...
- Thread running...
- Asking thread to stop...
- Thread interrupted...
- Thread exiting under request...
- Stopping application...
中断I/O操作
然而,如果线程在I/O操作进行时被阻塞,又会如何?I/O操作可以阻塞线程一段相当长的时间,特别是牵扯到网络应用时。例如,服务器可能需要等待一个请求(request),又或者,一个网络应用程序可能要等待远端主机的响应。
如果你正使用通道(channels)(这是在Java 1.4中引入的新的I/O API),那么被阻塞的线程将收到一个 ClosedByInterruptException异常。如果情况是这样,其代码的逻辑和第三个例子中的是一样的,只是异常不同而已。
但是,你可能正使用Java1.0之前就存在的传统的I/O,而且要求更多的工作。既然这样,Thread.interrupt()将不起作用,因为线 程将不会退出被阻塞状态。Listing D描述了这一行为。尽管interrupt()被调用,线程也不会退出被阻塞状态。
- Listing D
- import java.io.*;
- class Example4 extends Thread {
- public static void main( String args[] ) throws Exception {
- Example4 thread = new Example4();
- System.out.println( "Starting thread..." );
- thread.start();
- Thread.sleep( 3000 );
- System.out.println( "Interrupting thread..." );
- thread.interrupt();
- Thread.sleep( 3000 );
- System.out.println( "Stopping application..." );
- //System.exit( 0 );
- }
- public void run() {
- ServerSocket socket;
- try {
- socket = new ServerSocket(7856);
- } catch ( IOException e ) {
- System.out.println( "Could not create the socket..." );
- return;
- }
- while ( true ) {
- System.out.println( "Waiting for connection..." );
- try {
- Socket sock = socket.accept();
- } catch ( IOException e ) {
- System.out.println( "accept() failed or interrupted..." );
- }
- }
- }
- }
很幸运,Java平台为这种情形提供了一项解决方案,即调用阻塞该线程的套接字的close()方法。在这种情形下,如果线程被I/O操作阻 塞,该线程将接收到一个SocketException异常,这与使用interrupt()方法引起一个InterruptedException异常 被抛出非常相似。
唯一要说明的是,必须存在socket的引用(reference),只有这样close()方法才能被调用。这意味着socket对象必须被共享。Listing E描述了这一情形。运行逻辑和以前的示例是相同的。
- /*Listing E*/
- import java.net.*;
- import java.io.*;
- class Example5 extends Thread {
- volatile boolean stop = false;
- volatile ServerSocket socket;
- public static void main( String args[] ) throws Exception {
- Example5 thread = new Example5();
- System.out.println( "Starting thread..." );
- thread.start();
- Thread.sleep( 3000 );
- System.out.println( "Asking thread to stop..." );
- thread.stop = true;
- thread.socket.close();
- Thread.sleep( 3000 );
- System.out.println( "Stopping application..." );
- //System.exit( 0 );
- }
- public void run() {
- try {
- socket = new ServerSocket(7856);
- } catch ( IOException e ) {
- System.out.println( "Could not create the socket..." );
- return;
- }
- while ( !stop ) {
- System.out.println( "Waiting for connection..." );
- try {
- Socket sock = socket.accept();
- } catch ( IOException e ) {
- System.out.println( "accept() failed or interrupted..." );
- }
- }
- System.out.println( "Thread exiting under request..." );
- }
- }
以下是运行Listing E中代码后的输出:
- Starting thread...
- Waiting for connection...
- Asking thread to stop...
- accept() failed or interrupted...
- Thread exiting under request...
- Stopping application...
多线程是一个强大的工具,然而它正呈现出一系列难题。其中之一是如何中断一个正在运行的线程。如果恰当地实现,使用上述技术中断线程将比使用Java平台上已经提供的内嵌操作更为简单。
发表评论
-
Ja va 内 部 类 总结
2013-08-12 15:30 747这篇文章写的比较全面: http://lavasoft. ... -
一篇不错的讲解Java异常的文章
2013-08-12 12:46 800转载一篇比较老的文章,转自:http://www.blogj ... -
Java static class
2013-08-09 17:00 834讲的比较啰嗦,可以注意一下静态内部类的使用限制。 转自: ... -
单例模式的五种写法
2012-12-06 16:28 714转自:http://cantellow.iteye.co ... -
java 范型 约束
2012-12-03 10:20 3780转自: http://www.myexception. ... -
java 内部类
2012-12-02 16:48 726转自: http://www.cnblogs.com/s ... -
有条件的线程安全性和可伸缩性
2011-08-10 21:54 959转自:http://huangjiajian01. ... -
java同步包装器与线程安全
2011-08-10 21:24 4004同步线程包装器是为“原集合对象”的实际操作找一个代理对象 ...
相关推荐
线程的状态包括新建、运行、中断和死亡。线程的生命周期始于新建,通过调用start()方法进入就绪状态,然后可能进入运行、阻塞或等待,最后在run()方法执行完毕后死亡。 9.2 Thread的子类创建线程: 创建线程的一种...
- interrupt()方法用于中断线程,如果线程正在阻塞(如sleep或wait),会被中断并抛出InterruptedException。 通过这些技术,我们可以构建高效、稳定、响应迅速的多线程应用程序。在实际开发中,应根据具体需求...
本示例提供了可以直接运行的Java多线程代码,帮助开发者更好地理解和运用多线程。 一、继承Thread类 在Java中,可以通过创建一个新的类来继承Thread类,然后覆盖其run()方法,将需要并发执行的任务放入run()方法中...
9. **线程中断**:`interrupt()`方法可以标记线程中断状态,线程可以通过检查`isInterrupted()`或`interrupted()`方法来响应中断请求,从而优雅地停止线程执行。 10. **线程局部变量(ThreadLocal)**:为每个线程...
在Java编程中,控制程序执行超时是一项重要的任务,特别是在多线程环境下,我们可能需要确保某个任务不会无限制地运行下去,导致资源耗尽。本文将深入探讨如何使用Java的线程机制来实现程序执行的超时控制,同时也会...
本示例将详细探讨Java中断线程的正确方法,以确保线程安全且高效地退出。 首先,我们需要了解Java中的线程中断机制。线程中断是通过调用`Thread.interrupt()`方法来实现的,它会设置线程的中断标志。当线程正在运行...
Java提供了一个更安全的中断线程的机制,即`Thread.interrupt()`。当调用`interrupt()`时,目标线程的中断状态会被设置,线程在检查到中断状态后可以决定如何响应。例如,`Thread.sleep()`、`SocketInputStream....
4. **线程状态**:Java线程有五种状态:新建、就绪、运行、阻塞和终止。了解这些状态有助于理解线程的生命周期。 5. **同步机制**:Java提供了多种同步机制防止数据竞争,如`synchronized`关键字、`wait()`, `...
Java多线程编程是Java开发中的重要组成部分,它允许程序同时执行多个任务,极大地提高了程序的效率和响应性。在Java中,多线程主要通过继承Thread类或实现Runnable接口来实现。本教程《Java多线程编程核心技术》将...
以下是关于Java中断线程的一些关键知识点: 1. **线程状态**:在Java中,线程有多种状态,包括新建、可运行、运行、阻塞、等待、超时等待和终止。中断线程主要是针对那些处于运行、阻塞或等待状态的线程。 2. **...
而检查异常(Checked Exceptions)如果在线程中抛出,需在该线程或其祖先线程中捕获,否则会导致线程中断。 最后,书中还将涵盖Java内存模型(JMM)和volatile关键字。JMM定义了线程如何访问共享变量的规则,保证了...
Java如何中断一个正在运行的线程 Java中的线程中断是指在某个线程完成任务之前停止其正在进行的...本文对Java中断一个正在运行的线程进行了详细的介绍和分析,旨在帮助读者更好地理解和掌握线程中断的知识和技能。
Java中的线程有五种状态:新建(New)、就绪(Runnable)、运行(Running)、等待/阻塞(Blocked)、终止(Terminated)。了解这些状态以及它们之间的转换对于理解多线程的执行流程非常重要。 三、线程同步 1. ...
5. **线程状态**:Java线程有五种基本状态:新建、可运行、运行、阻塞和死亡。线程的状态转换反映了其生命周期的不同阶段。 6. **线程同步**:为了避免线程间的冲突,Java提供了多种同步机制,如`synchronized`...
当run()方法执行完毕或被中断,线程死亡。 三、线程同步与互斥 为了解决多个线程访问共享资源时可能出现的数据不一致问题,Java提供了synchronized关键字来实现线程同步。synchronized可以修饰方法或代码块,确保...
在 Java 中,线程中断是通过 interrupt 方法来实现的,该方法只是给线程置一个中断标志位,并不会立即停止线程的运行。interrupted 方法可以判断是否被中断,静态方法 interrupted() 则可以判断是否被中断,并清除...
Java并发多线程是Java编程中的重要组成部分,它允许程序同时执行多个任务,极大地提高了程序的效率和响应性。在Java中,多线程主要通过`Thread`类、`Runnable`接口以及`ExecutorService`来实现。下面我们将深入探讨...
在Java编程中,多线程是并发执行任务的重要方式,然而在实际应用中,我们可能会遇到某些线程执行时间过长或死锁的情况,这可能导致系统资源的浪费甚至整体性能下降。因此,对线程进行超时监控是必要的,以确保程序的...
Java提供了一些线程控制方法,如`sleep()`使线程休眠,`join()`等待线程结束,`yield()`让当前线程暂停,以及`interrupt()`中断线程。 同步机制防止多个线程同时访问共享资源,Java提供了以下几种方式: 1. `...
- **简化信号处理**:线程可以在接收到外部信号时快速响应,无需中断整个进程。 - **支持分布式对象**:多线程使得跨网络的数据交换更加高效。 - **跨平台编译**:如POSIX线程和Java线程,可以在不同平台上使用相同...