一步一步掌握java的线程机制(二)----Thread的生命周期

2019-03-12 17:19|来源: 网络

      之前讲到Thread的创建,那是Thread生命周期的第一步,其后就是通过start()方法来启动Thread,它会执行一些内部的管理工作然后调用Thread的run()方法,此时该Thread就是alive(活跃)的,而且我们还可以通过isAlive()方法来确定该线程是否启动还是终结。

      一旦启动Thread后,我们就只能执行一个方法:run(),而run()方法就是负责执行Thread的任务,所以终结Thread的方法很简单,就是终结run()方法。仔细查看文档,我们会发现里面有一个方法:stop(),似乎可以用来停止Thread,但是这个方法已经被废除了,因为它存在着内部的竞争。

      我们经常需要一个不断执行的Thread,然后在某个特定的条件下才会终结它,方法有很多,但最常用的有设定标记和中断Thread两种方式。

      我们将之前例子中的Thread改写一下:

public class RandomCharacterGenerator extends Thread implements CharacterSource {
    static char[] chars;
    static String charArray = "abcdefghijklmnopqrstuvwxyz0123456789";
    static {
        chars = charArray.toCharArray();
    }
    
    private volatile boolean done = false;

    Random random;
    CharacterEventHandler handler;

    public RandomCharacterGenerator() {
        random = new Random();
        handler = new CharacterEventHandler();
    }

    public int getPauseTime() {
        return (int) (Math.max(1000, 5000 * random.nextDouble()));
    }

    @Override
    public void addCharacterListener(CharacterListener cl) {
        handler.addCharacterListener(cl);
    }

    @Override
    public void removeCharacterListener(CharacterListener cl) {
        handler.removeCharacterListener(cl);
    }

    @Override
    public void nextCharacter() {
        handler.fireNewCharacter(this,
                (int) chars[random.nextInt(chars.length)]);
    }

    public void run() {
        while(!done){
            nextCharacter();
            try {
                Thread.sleep(getPauseTime());
            } catch (InterruptedException ie) {
                return;
            }
        }
    }
    
    public void setDone(){
        done = true;
    }
}

      现在我们多了一个标记:done,这样我们就可以在代码中通过调用setDone()来决定什么时候停止该Thread。这里使用了volatile关键字,它主要是为了同步。这点会放在同步这里讲。
      设定标记的最大问题就是我们必须等待标记的状态,这样就会造成延迟。当然,这种延迟是无法避免的,但必须想办法缩短到最小。于是,中断Thread这种方法就有它的发挥地方了。

      我们可以通过interrupt()方法来中断Thread,该方法会造成两个副作用:

1.它会导致任何的阻塞方法都会抛出InterruptedException,我们必须强制性的捕获这个错误哪怕我们根本就不需要处理它,这也是java的异常处理机制让人诟病的一个地方。

2.设定Thread对象内部的标记来指示此Thread已经被中断了,像是这样:

public void run(){
     while(!isInterrupted()){
        ...
      }
}

      虽然无法避免延迟,但是延迟已经被缩短了。
      无论是采用标记还是中断的方法,我们之所以无法消除延迟的原因是我们无法确定是检查标记先还是调用方法先,这就是所谓的race condition,是线程处理中永远无法避免的话题。

      Thread不仅可以被终结,还可以暂停,挂起和恢复。

      Thread原本有suspend()方法和resume()方法来执行挂起和恢复,但它们和stop()出于同样的原因,都被废除了。

      我们可以通过sleep()方法来挂起Thread,当在指定的时间后,它就会自动恢复。严格意义上讲,sleep并不等同于suspend,真正的suspend应该是由一个线程来挂起另一个线程,但是sleep只会影响当前的Thread。要想真正实现挂起和恢复,我们可以使用等待和通知机制,但这个机制最大的问题就是我们的Thread必须使用该技术来编写。

      Thread在终结后,如果有可能,我们还需要对它进行善后。即使Thread已经被终结了,但是其他对象只要还持有它的引用,它们就可以调用该Thread的资源,这也会导致该Thread无法被回收。

       但我们有时候还是希望继续保持该Thread的引用,因为我们想要判别它是否真的已经完成了工作,可以使用join()方法。join()方法会被阻塞住直到Thread完成它的run()方法,但是这个存在风险:第一次对join()方法的调用可能会一直被阻塞住很长时间直到Thread真正完成,所以,一般情况下我们还是使用isAlive()方法来判断。

       由于我们可以通过实现一个Runnable接口来定义我们的任务,所以在判断所在线程是否已经中断的时候,就有一个问题:该任务还没有绑定到任何线程上。我们可以通过currentThread()方法来获得当前Thread的引用,接着调用isInterrupted()来判断线程是否中断。

      

     

相关问答

更多
  • 兄弟连JavaEE战狼班: 第一阶段:Java语言基础 ★ Java语言基础 1、面向对象思维JAVASE 2、(类加载机制与反射,annotation,泛型,网络编程,多线程,IO,异常处理,常用API,面向对象,JAVA编程基础) 3、Java8新特性 第二阶段:数据库 ★ 数据库 1、Oracle(SQL语句、SQL语句原理、SQL语句优化、表、视图 2、序列、索引、Oracle数据字典、Oracle 数据库PL/SQL开发 3、数据库设计原则、 MySQL 、 JDBC 第三阶段:Web基础 ★ W ...
  • 其实Java学的好坏,一方面是理论和语法,一方面就是熟悉API,Java就是因为有一个强大的类库,所以你光有编程思想,光会了语法,还不行,还得熟悉这些Sun提供的类,但是挨个看API一个个方法读也不是办法,而且也不用什么都记,那么多类挨个记,记得住么?不能这么学,分清主次,一些重要的类,常用的类,常用的方法有些记住,比如说System.out.print要是还去查文档那就太过份了,有些熟悉就行,剩下的用到就查文档,我们做Java的,这辈子只要是还干Java,这个API文档就离不开了,什么方法不明白马上去查, ...
  • 进入这个网址 http://www.verycd.com/topics/2767912/,把韩顺平老师的视频教程下下来学,很不错
  • 文件没保存吧,难道你都是手动敲的?eclipse有自动填充的
  • 你好你可以看“王鹰豪二阶魔方视频”里面有详细易懂的还原讲解视频,王鹰豪是二阶魔方中国记录保持者,最强大脑第三季队长
  • 先画身体,壮壮的! 还有‘恐龙'腿!要画粗一点哦。 紧跟着是尾巴!长一点。 添上眼睛和鼻子! 然后就是身上很有特色的恐龙背刺啦! 最后画上一棵树!不知它吃不吃树叶!
  • 硬件:先学每个硬件的作用.电路接口.组成.结构. 软件:选学软件如何使用.设定.修改.编辑.制做. 网络.先学网络如何使用.作用.构成.编辑.修改.制做.破解 图形图像:先学绘画基本功.专业名词.专业术语.临摸.修改.设计.创作.
  • run()方法调用B构造函数然后返回。 所以这是在生成的线程中发生的唯一事情:执行B的构造函数。 在构造函数中运行长进程是一种严重的设计气味。 构造函数并不意味着执行一长串指令。 它们意味着构建一个对象。 代码应该更像下面的代码: Thread t = new Thread(new Runnable() { @Override public void run() { B b = new B(); b.doSomethingLong(); } }); ...
  • 有一定的数据库基础吗?没有的话,建议从SQL语句学起。比较好的教材是Oracle OCP认证的《SQL and PL/SQL》。学习SQL的时候,尽可能坚持使用Oracle自带的工具:SQLPLUS。  有了一定的SQL基础后,就要尽可能的了解Oracle的体系结构,这就涉及到了Oracle管理的内容了。《Oracle10g OCP认证手册》这本书不错。 不过,如果是初学者的话,不建议自己去摸索,因为这样往往会如盲人摸象,不仅会事倍功半,而且会有一些错误的概念。你可以去CUUG报个DBA就业培训班进行学习, ...
  • 将这些内容放入Tomcat服务器中可能更容易 - 并且跨平台。 查看Quartz调度程序 ,它完成了cron所做的事情,更容易集成到Web应用程序中,并允许您调用用Java编写的作业,而无需任何复杂的管道。 作为一个额外的优势,这些作业可以像Tomcat中的任何其他Web应用程序一样进行部署。 It's probably easier - and cross platform - to just put that stuff inside your Tomcat server. Check out the ...