zoukankan      html  css  js  c++  java
  • java线程中断[interrupt()函数] (转载)

    一个正常的线程中断: 从运行到真正的结束,应该有三个阶段: 正常运行. 处理结束前的工作,也就是准备结束. 结束退出.
    Java曾经提供过抢占式限制中断,但问题多多,例如的Thread.stop。另一方面,出于Java应用代码的健壮性的考虑,降低了编程门槛,减少不清楚底层机制的程序员无意破坏系统的概率,这个问题很多,比如:
      当在一个线程对象上调用stop()方法时,这个线程对象所运行的线程就会立即停止,并抛出特殊的ThreadDeath()异常。这里的“立即”因为太“立即”了,
      一个线程正在执行:

    Java代码 复制代码 收藏代码
    1. synchronized void { x = 3; y = 4;}   
    synchronized void { x = 3; y = 4;} 

     
    由于方法是同步的,多个线程访问时总能保证x,y被同时赋值,而如果一个线程正在执行到x = 3;时,被调用了 stop()方法,即使在同步块中,它也干脆地stop了,这样就产生了不完整的残废数据。而多线程编程中最最基础的条件要保证数据的完整性,所以请忘记 线程的stop方法,以后我们再也不要说“停止线程”了。
       如何才能“结束”一个线程?
        如今,Java的线程调度不提供抢占式中断,而采用协作式的中断。其实,协作式的中断,原理很简单,就是轮询某个表示中断的标记,我们在任何普通代码的中都可以实现。 例如下面的代码:

    Java代码 复制代码 收藏代码
    1. volatile bool isInterrupted;   
    2.   
    3. //…   
    4.   
    5. while(!isInterrupted) {   
    6.   
    7.     compute();   
    8.   
    9. }  
        volatile bool isInterrupted;
    
        //…
    
        while(!isInterrupted) {
    
            compute();
    
        }

    interrupt就是这样的一个通知,将Thead里的中断标志位设为true,而线程能否退出,就看用户的代码对于这个通知是怎么处理的了。
    对于处于sleep,join等操作的线程,如果被调用interrupt()后,会抛出InterruptedException,然后线程的中断标志位会由true重置为false,因为线程为了处理异常已经重新处于就绪状态。
    我在运行thinking in java里中断的例子时,一直都很奇怪为什么在catch(InterruptedException e)的处理段里,thead.isInterrupted()返回的都是false,原来是已被重置,所以很多导致无法中断线程,原因是出在这里。例如这段代码:

    Java代码 复制代码 收藏代码
    1. public class ThreadA extends Thread{   
    2.    int count=0;   
    3.       
    4.    public void run(){   
    5.        System.out.println(getName()+"将要运行...");   
    6.        while(!this.isInterrupted()){   
    7.            System.out.println(getName()+"运行中"+count++);   
    8.            try{   
    9.                Thread.sleep(400);   
    10.            }catch(InterruptedException e){   
    11.                System.out.println(getName()+"从阻塞中退出...");   
    12.                System.out.println("this.isInterrupted()="+this.isInterrupted());   
    13.   
    14.            }   
    15.        }   
    16.        System.out.println(getName()+"已经终止!");   
    17.    }   
    18. }  
    public class ThreadA extends Thread{
       int count=0;
       
       public void run(){
           System.out.println(getName()+"将要运行...");
           while(!this.isInterrupted()){
               System.out.println(getName()+"运行中"+count++);
               try{
                   Thread.sleep(400);
               }catch(InterruptedException e){
                   System.out.println(getName()+"从阻塞中退出...");
                   System.out.println("this.isInterrupted()="+this.isInterrupted());
    
               }
           }
           System.out.println(getName()+"已经终止!");
       }
    }
    Java代码 复制代码 收藏代码
    1. public class ThreadDemo{   
    2.        
    3.     public static void main(String argv[])throws InterruptedException{   
    4.         ThreadA ta=new ThreadA();   
    5.         ta.setName("ThreadA");   
    6.         ta.start();   
    7.         Thread.sleep(2000);   
    8.         System.out.println(ta.getName()+"正在被中断...");   
    9.         ta.interrupt();   
    10.         System.out.println("ta.isInterrupted()="+ta.isInterrupted());   
    11.     }   
    12.   
    13. }  
    public class ThreadDemo{
        
        public static void main(String argv[])throws InterruptedException{
            ThreadA ta=new ThreadA();
            ta.setName("ThreadA");
            ta.start();
            Thread.sleep(2000);
            System.out.println(ta.getName()+"正在被中断...");
            ta.interrupt();
            System.out.println("ta.isInterrupted()="+ta.isInterrupted());
        }
    
    }

    这段代码ThreadA线程永远都无法中断。
        实际上,JVM内部确实为每个线程维护了一个中断标记。但应用程序不能直接访问这个中断变量,必须通过下面几个方法进行操作:

    Java代码 复制代码 收藏代码
    1. public class Thread {   
    2.     //设置中断标记   
    3.     public void interrupt() { ... }     
    4.     //获取中断标记的值   
    5.     public boolean isInterrupted() { ... }   
    6.     //清除中断标记,并返回上一次中断标记的值   
    7.     public static boolean interrupted() { ... }      
    8.     ...   
    9. }  
        public class Thread {
            //设置中断标记
            public void interrupt() { ... }  
            //获取中断标记的值
            public boolean isInterrupted() { ... }
            //清除中断标记,并返回上一次中断标记的值
            public static boolean interrupted() { ... }   
            ...
        }

    如何使用中断标记来结束你的程序就是你自己来考虑的事了,事实上JVM只为我们设计一个中断标记而已。

  • 相关阅读:
    OpenGL(九) 三维混色和深度缓存设置
    自由度(degree of freedom)
    自由度(degree of freedom)
    非参贝叶斯(Bayesian Non-parameter)初步
    非参贝叶斯(Bayesian Non-parameter)初步
    一个GCC4.6.3的奇妙问题的糊涂解决方案
    Rational Rose--简介
    android 国际化
    日志文件C++ 时间 文件 行数
    看原理图之UART
  • 原文地址:https://www.cnblogs.com/xiaoerlang/p/3409505.html
Copyright © 2011-2022 走看看