Java怎样中断一个运行中的线程(1)

来源:互联网 发布:linux同步到windows 编辑:程序博客网 时间:2024/06/03 16:37

    编程人员在编程过程中,多线程呈现出了一组新的难题,而如何中断一个正在运行的线程则是难题之一。本文中,作者将讲述相关的方法。

     

    程序是很简易的。然而,在编程人员面前,多线程呈现出了一组新的难题,如果没有被恰当的解决,将导致意外的行为以及细微的、难以发现的错误。

    在本篇文章中,我们针对这些难题之一:如何中断一个正在运行的线程。

    背景中断(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 Bclass 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()时,这里仅举出一些。

    他们都可能永久的阻塞线程。即使发生超时,在超时期满之前持续等待也是不可行和不适当的,所以,要使用某种机制使得线程更早地退出被阻塞的状态。

    很不幸运,不存在这样一种机制对所有的情况都适用,但是,根据情况不同却可以使用特定的技术。在下面的环节,我将解答一下最普遍的例子。