`
bulote
  • 浏览: 1305847 次
文章分类
社区版块
存档分类
最新评论

收集的中断线程

 
阅读更多


<meta content="text/html; charset=utf-8" http-equiv="CONTENT-TYPE">

<meta content="OpenOffice.org 3.2 (Linux)" name="GENERATOR"> <style type="text/css"> <!-- @page { margin: 2cm } P { margin-bottom: 0.21cm } --> </style>

在JAVA中,通过其对线程类的内嵌支持,编程人员编写多线程程序是很简易的。然而,在编程人员面前,多线程呈现出了一组新的难题,如果没有被恰当的解 决,将导致意外的行为以及细微的、难以发现的错误。
在本篇文章中,我们针对这些难题之一:如何中断一个正在运行的线程。

背景
中断(Interrupt)一个线程意味着在该线程完成任务之前停止其正在进行的一切,有效地中止其当前的操作。线程是死亡、还是等待新的任务或是继续运 行至下一步,就取决于这个程序。虽然初次看来它可能显得简单,但是,你必须进行一些预警以实现期望的结果。你最好还是牢记以下的几点告诫。

首先,忘掉Thread.stop方法。虽然它确实停止了一个正在运行的线程,然而,这种方法是不安全也是不受提倡的,这意味着,在未来的JAVA版本 中,它将不复存在。

一些轻率的家伙可能被另一种方法Thread.interrupt所迷惑。尽管,其名称似乎在暗示着什么,然而,这种方法并不会中断一个正在运行的线程 (待会将进一步说明),正如ListingA中描述的那样。它创建了一个线程,并且试图使用Thread.interrupt方法停止该线程。 Thread.sleep()方法的调用,为线程的初始化和中止提供了充裕的时间。线程本身并不参与任何有用的操作。

ListingA

java 代码

    classExample1extendsThread{

    booleanstop=false;

    publicstaticvoidmain(Stringargs[])throwsException{

    Example1thread=newExample1();

    System.out.println("Startingthread...");

    thread.start();

    Thread.sleep(3000);

    System.out.println("Interruptingthread...");

    thread.interrupt();

    Thread.sleep(3000);

    System.out.println("Stoppingapplication...");

    //System.exit(0);

    }

    publicvoidrun(){

    while(!stop){

    System.out.println("Threadisrunning...");

    longtime=System.currentTimeMillis();

    while((System.currentTimeMillis()-time<1000)){

    }

    }

    System.out.println("Threadexitingunderrequest...");

    }

    }

如果你运行了ListingA中的代码,你将在控制台看到以下输出:

Startingthread...

Threadisrunning...

Threadisrunning...

Threadisrunning...

Interruptingthread...

Threadisrunning...

Threadisrunning...

Threadisrunning...

Stoppingapplication...

Threadisrunning...

Threadisrunning...

Threadisrunning...
...............................
甚至,在Thread.interrupt()被调用后,线程仍然继续运行。

真正地中断一个线程

中断线程最好的,最受推荐的方式是,使用共享变量(sharedvariable)发出信号,告诉线程必须停止正在运行的任务。线程必须周期性 的核查这一变量(尤其在冗余操作期间),然后有秩序地中止任务。ListingB描述了这一方式。

ListingB

java 代码

    classExample2extendsThread{

    volatilebooleanstop=false;

    publicstaticvoidmain(Stringargs[])throwsException{

    Example2thread=newExample2();

    System.out.println("Startingthread...");

    thread.start();

    Thread.sleep(3000);

    System.out.println("Askingthreadtostop...");

    thread.stop=true;

    Thread.sleep(3000);

    System.out.println("Stoppingapplication...");

    //System.exit(0);

    }

    publicvoidrun(){

    while(!stop){

    System.out.println("Threadisrunning...");

    longtime=System.currentTimeMillis();

    while((System.currentTimeMillis()-time<1000)&&(!stop)){

    }

    }

    System.out.println("Threadexitingunderrequest...");

    }

    }

运行ListingB中的代码将产生如下输出(注意线程是如何有秩序的退出的)

Startingthread...

Threadisrunning...

Threadisrunning...

Threadisrunning...

Askingthreadtostop...

Threadexitingunderrequest...

Stoppingapplication...

虽然该方法要求一些编码,但并不难实现。同时,它给予线程机会进行必要的清理工作,这在任何一个多线程应用程序中都是绝对需要的。请确认将共享变量 定义成volatile类型或将对它的一切访问封入同步的块/方法(synchronizedblocks/methods)中。

到目前为止一切顺利!但是,当线程等待某些事件发生而被阻塞,又会发生什么?当然,如果线程被阻塞,它便不能核查共享变量,也就不能停止。这在许多情况下 会发生,例如调用Object.wait()、ServerSocket.accept()和DatagramSocket.receive()时,这里 仅举出一些。

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

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

使用Thread.interrupt()中断线程

正如ListingA中所描述的,Thread.interrupt()方法不会中断一个正在运行的线程。这一方法实际上完成的是,在线程受到阻塞 时抛出一个中断信号,这样线程就得以退出阻塞的状态。更确切的说,如果线程被Object.wait,Thread.join和 Thread.sleep三种方法之一阻塞,那么,它将接收到一个中断异常(InterruptedException),从而提早地终结被阻塞状态。

因此,如果线程被上述几种方法阻塞,正确的停止线程方式是设置共享变量,并调用interrupt()(注意变量应该先设置)。如果线程没有被阻塞,这时调用interrupt()将不起作用;否则,线 程就将得到异常(该线程必须事先预备好处理此状况),接着逃离阻塞状态。在任何一种情况中,最后线程都将检查共享变量然后再 停止。ListingC这个示例描述了该技术。

ListingC

java 代码

java 代码

    classExample3extendsThread{

    volatilebooleanstop=false;

    publicstaticvoidmain(Stringargs[])throwsException{

    Example3thread=newExample3();

    System.out.println("Startingthread...");

    thread.start();

    Thread.sleep(3000);

    System.out.println("Askingthreadtostop...");

    thread.stop=true;//如果线程阻塞, 将不会检查此变量

    thread.interrupt();

    Thread.sleep(3000);

    System.out.println("Stoppingapplication...");

    //System.exit(0);

    }

    publicvoidrun(){

    while(!stop){

    System.out.println("Threadrunning...");

    try{

    Thread.sleep(1000);

    }catch(InterruptedExceptione){

    System.out.println("Threadinterrupted...");

    }

    }

    System.out.println("Threadexitingunderrequest...");

    }

    }

一旦ListingC中的Thread.interrupt()被调用,线程便收到一个异常,于是逃离了阻塞状态并确定应该停止。运行以上代码将 得到下面的输出:

Startingthread...

Threadrunning...

Threadrunning...

Threadrunning...

Askingthreadtostop...

Threadinterrupted...

Threadexitingunderrequest...

Stoppingapplication...


中断I/O操作
然而,如果线程在I/O操作进行时被阻塞,又会如何?I/O操作可以阻塞线程一段相当长的时间,特别是牵扯到网络应用时。例如,服务器可能需要等 待一个请求(request),又或者,一个网络应用程序可能要等待远端主机的响应。

如果你正使用通道(channels)(这是在Java1.4中引入的新的I/OAPI),那么被阻塞的线程将收到一个 ClosedByInterruptException异常。如果情况是这样,其代码的逻辑和第三个例子中的是一样的,只是异常不同而已。

但是,你可能正使用Java1.0之前就存在的传统的I/O,而且要求更多的工作。既然这样,Thread.interrupt()将不起作用,因为线程 将不会退出被阻塞状态。ListingD描述了这一行为。尽管interrupt()被调用,线程也不会退出被阻塞状态

ListingD

java 代码

    importjava.io.*;

    classExample4extendsThread{

    publicstaticvoidmain(Stringargs[])throwsException{

    Example4thread=newExample4();

    System.out.println("Startingthread...");

    thread.start();

    Thread.sleep(3000);

    System.out.println("Interruptingthread...");

    thread.interrupt();

    Thread.sleep(3000);

    System.out.println("Stoppingapplication...");

    //System.exit(0);

    }

    publicvoidrun(){

    ServerSocketsocket;

    try{

    socket=newServerSocket(7856);

    }catch(IOExceptione){

    System.out.println("Couldnotcreatethesocket...");

    return;

    }

    while(true){

    System.out.println("Waitingforconnection...");

    try{

    Socketsock=socket.accept();

    }catch(IOExceptione){

    System.out.println("accept()failedorinterrupted...");

    }

    }

    }

    }

很幸运,Java平台为这种情形提供了一项解决方案,即调用阻塞该线程的套接字的close()方法。在这种情形下,如果线程被I/O操作阻塞,该线程将 接收到一个SocketException异常,这与使用interrupt()方法引起一个InterruptedException异常被抛出非常相 似。

唯一要说明的是,必须存在socket的引用(reference),只有这样close()方法才能被调用。这意味着socket对象必须被共享。 ListingE描述了这一情形。运行逻辑和以前的示例是相同的。

ListingE

java 代码

    importjava.net.*;

    importjava.io.*;

    classExample5extendsThread{

    volatilebooleanstop=false;

    volatileServerSocketsocket;

    publicstaticvoidmain(Stringargs[])throwsException{

    Example5thread=newExample5();

    System.out.println("Startingthread...");

    thread.start();

    Thread.sleep(3000);

    System.out.println("Askingthreadtostop...");

    thread.stop=true;

    thread.socket.close();

    Thread.sleep(3000);

    System.out.println("Stoppingapplication...");

    //System.exit(0);

    }

    publicvoidrun(){

    try{

    socket=newServerSocket(7856);

    }catch(IOExceptione){

    System.out.println("Couldnotcreatethesocket...");

    return;

    }

    while(!stop){

    System.out.println("Waitingforconnection...");

    try{

    Socketsock=socket.accept();

    }catch(IOExceptione){

    System.out.println("accept()failedorinterrupted...");

    }

    }

    System.out.println("Threadexitingunderrequest...");

    }

    }

以下是运行ListingE中代码后的输出:

Startingthread...

Waitingforconnection...

Askingthreadtostop...

accept()failedorinterrupted...

Threadexitingunderrequest...

Stoppingapplication...

多线程是一个强大的工具,然而它正呈现出一系列难题。其中之一是如何中断一个正在运行的线程。如果恰当地实现,使用上述技术中断线程将比使用Java平台 上已经提供的内嵌操作更为简单。

分享到:
评论

相关推荐

    09.多线程编程基础-停止线程-使用interrupt方法中断线程.mp4

    在学习Java过程中,自己收集了很多的Java的学习资料,分享给大家,有需要的欢迎下载,希望对大家有用,一起学习,一起进步。

    java8看不到源码-awesome-articles:不同主题的文章精选列表

    中中断线程 - ExecutorService 解释 - ExecutorService 风险 - Executor Service 10 提示和技巧 和 - 线程解释,线程状态是什么以及如何分析线程转储。 - 通过分步指南及其背后的原因解决真正复杂的问题。 Maven - ...

    Java并发编程实战

    4.1.1 收集同步需求47 4.1.2 依赖状态的操作48 4.1.3 状态的所有权48 4.2 实例封闭49 4.2.1 Java监视器模式51 4.2.2 示例:车辆追踪51 4.3 线程安全性的委托53 4.3.1 示例:基于委托的车辆追踪器54 4.3.2 ...

    resin-jvm 调优

    尽管这种收集器可以消除中断,但是收集器需花费较长的时间寻找死对象,而且处理应用程序时收集器经常运行。如果处理器不能应付应用程序产生的垃圾,它会中断应用程序并关闭收集。 分代并发收集器 这种收集器在护理...

    Java 并发编程实战

    4.1.1 收集同步需求 4.1.2 依赖状态的操作 4.1.3 状态的所有权 4.2 实例封闭 4.2.1 Java监视器模式 4.2.2 示例:车辆追踪 4.3 线程安全性的委托 4.3.1 示例:基于委托的车辆追踪器 4.3.2 独立的状态变量 ...

    linux操作系统大全

    我精心收集的十分详细的linux教程! linux操作系统分析 1.linux简介 2.常用命令 3.网络功能 4.系统管理 5.外壳编程 6.gawk语言编程 7.perl语言编程 8.linux内核简介 9.系统进程 10.内存管理 11.进程通信 12.PCI 13....

    Python3进程内存分析器-用于运行python进程的内存分析器.zip

    尽管内存分析器收集有关内存中对象的信息时您的过程(及其所有线程!)将被暂停,但是运行分析不会(不会咳嗽)中断您的过程。 除了requirements*.txt运行前端的deps之外,还需要在目标二进制文件的库路径中安装...

    Android---课程设计

    2.计算收集时长(单位:ms) 3.连续计时(允许中断) 4.秒表 5.启动/关闭键(绿表停,红表运行) 6.采集的数据以文件形式保存,路径:storage/sdcard0/222.txt 技术: 1.UI界面 2.传感器管理器API 3.多线程 4.文件...

    双嘉邮件地址搜索联盟 5.1.0.1.rar

    10、以项目为搜索单位,停止搜索时,收集状态自动保存,你可以随时中断收集并保存工作状态,下次可以打开对应项目名称,从停止点开始继续这个工作; 11、自动检查重复邮箱和不合格邮箱并即时过滤; 12、以项目为...

    java面试常见基础(深层次,高级研发)

    Cache怎么实现的 fixed线程如果中断,线程回自己销毁么? 51 23. 栅栏的原理和实现。 51 23.1. 1. CyclicBarrier简介 51 23.2. 2. CyclicBarrier数据结构 52 23.3. 3. CyclicBarrier源码分析(基于JDK1.7.0_40) 52 ...

    思达电子邮箱地址搜索软件Email Spider 9.1

    7、收集状态自动保存,你也可以随时中断收集并保存工作状态,下次可以打开工作,从停止点开始继续这个工作; 8、自动检查重复邮箱和不合格邮箱并即时删除; 9、邮箱的批量导入导出,支持文本、EXCEL、FOXPFO、Access...

    电子邮件机器人V9.8

    7、收集状态自动保存,你也可以随时中断收集并保存工作状态,下次可以打开工作,从停止点开始继续这个工作; 8、自动检查重复邮箱和不合格邮箱并即时删除; 9、邮箱的批量导入导出,支持文本、EXCEL、FOXPFO、Access...

    清华大学Linux操作系统原理与应用

    6.5 实例——利用系统调用实现一个调用日志收集系统 143 6.5.1 代码体系结构 143 6.5.2 把代码集成到内核中 146 6.5.3 实现步骤 148 习题6 148 第7章 内核中的同步 149 7.1 临界区和竞争状态 149 7.1.1 临界区举例 ...

    双嘉邮件地址搜索联盟 V 5.1.0.1

    10、以项目为搜索单位,停止搜索时,收集状态自动保存,你可以随时中断收集并保存工作状态,下次可以打开对应项目名称,从停止点开始继续这个工作; 11、自动检查重复邮箱和不合格邮箱并即时过滤; 12、以项目为...

    基于Java的即时通讯系统设计与实现【文献综述】.doc

    Java程序可以有多个执行线程,如可以让一个线程进行复杂的计算,而让另一个线程 与用户进行交互,这样用户可以在不中断计算线程的前提下与系统进行交互。多线程保 证了较高的执行效率。 (4) 分布性 Java是面向网络...

    java初学者必看

    1.8 垃圾收集器 1.9 本章习题 第2章 Java开发环境 2.1 J2SE的下载和安装 2.1.1 J2SE的下载 2.1.2 J2SE的安装 2.2 环境变量的配置与测试 2.2.1 设置环境变量path 2.2.2 设置环境变量classpath 2.2.3 环境...

    fastsum:一种尽快计算大型目录的安全校验和的工具

    该代码还一次只允许一个线程读取大文件,前提是顺序读取一个大文件比中断在别处的按顺序读取要快。 (就fastsum而言,“大文件”的大小超过256 kB。您可以通过-b参数指定限制,可接受的后缀为“ k”和“ M”。) ...

    UNIX 高级教程系统技术内幕

    12.8.1 垃圾收集 12.8.2 分析 12.9 多处理器的分层分配器 12.9.1 分析 12.10 Solaris 2.4 的Slab 分配器 12.10.1 对象复用 12.10.2 硬件Cache 利用率 12.10.3 分配器footprint 12.10.4 设计与接口 12.10.5 实现 ...

    Erlang安装手册

    它的运行时系统甚至允许代码在不被中断的情况下更新。另外如果你需要更高效的话,字节代码也可以编译成本地代码运行。  Erlang特性:  ● 并发性 - Erlang支持超大量级的并发线程,并且不需要操作系统具有并发...

Global site tag (gtag.js) - Google Analytics