• 线程的几种状态以及sleep/wait/yield/join的区别


    一.线程通常有五种状态:

    1. 新建状态(New):新创建了一个线程对象。
    2. 就绪状态(Runnable):线程对象创建后,其他线程调用了该对象的start()方法。该状态的线程位于可运行线程池中,变得可运行,等待 获取CPU的使用权。
    3. 运行状态(Running):就绪状态的线程获取了CPU,执行程序代码。
    4. 阻塞状态(Blocked):阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进入就绪状态,才有机会转到运行状态。
    5. 5.死亡状态(Dead):线程执行完了或者因异常退出了run()方法,该线程结束生命周期。

    二.阻塞的情况分为三种:

    1. 等待阻塞:运行的线程执行wait()方法,该线程会释放占用的所有资源,JVM会把该线程放入“等待池”中。进入这个状态后,是不能自动唤醒的,必须依靠其他线程调用notify()或notifyAll()方法才能被唤醒,wait是object类的方法
    2. 同步阻塞:运行的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入“锁池”中。
    3. 其他阻塞:运行的线程执行sleep()或join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。sleep是Thread类的方法。

    三.Java中sleep,wait,yield,join的区别

    1. sleep()方法
    • 在指定时间内让当前正在执行的线程暂停执行,但不会释放“锁标志”.不推荐使用.
    • sleep()使当前线程进入阻塞状态,在指定时间内不会执行。
    1. wait()方法
    • 在其他线程调用对象的notify或notifyAll方法前,导致当前线程等待。线程会释放掉它所占有的“锁标志”,从而使别的线程有机会抢占该锁。
    • 当前线程必须拥有当前对象锁。如果当前线程不是此锁的拥有者,会抛出IllegalMonitorStateException异常。
    • 唤醒当前对象锁的等待线程使用notify或notifyAll方法,也必须拥有相同的对象锁,否则也会抛出IllegalMonitorStateException异常。
    • waite()和notify()必须在synchronized函数或synchronized block中进行调用。如果在non-synchronized函数或non-synchronized block中进行调用,虽然能编译通过,但在运行时会发生IllegalMonitorStateException的异常。
    1. yield方法
    • 暂停当前正在执行的线程对象。
    • yield()只是使当前线程重新回到可执行状态,所以执行yield()的线程有可能在进入到可执行状态后马上又被执行。
    • yield()只能使同优先级或更高优先级的线程有执行的机会。
    • 调用yield方法并不会让线程进入阻塞状态,而是让线程重回就绪状态,它只需要等待重新获取CPU执行时间,这一点是和sleep方法不一样的。
    1. join方法
    • 等待该线程终止。
    • 等待调用join方法的线程结束,再继续执行。如:t.join();//主要用于等待t线程运行结束,若无此句,main则会执行完毕,导致结果不可预测。
    • 在很多情况下,主线程创建并启动了线程,如果子线程中药进行大量耗时运算,主线程往往将早于子线程结束之前结束。这时,如果主线程想等待子线程执行完成之后再结束,比如子线程处理一个数据,主线程要取得这个数据中的值,就要用到join()方法了。方法join()的作用是等待线程对象销毁。

    注意:sleep()方法给其他线程运行机会时不考虑线程的优先级,因此会给低优先级的线程以运行的机会;yield()方法只会给相同优先级或更高优先级的线程以运行的机会
    join的意思是会等到调用改join方法的线程执行完毕之后才会执行其他线程,举例说明:

    public static void main(String[] args) throws InterruptedException {
    //开启一个线程
    Thread thread = new Thread(new Runnable() {
    @Override
    public void run() {
    System.out.println("我是新开副线程的执行");
    //这个方法会先让副线程沉睡1秒之后在执行,但是期间不会影响主线程的执行
    Thread.sleep(1000L);
    }
    });
    thread.start();//启动线程
    //这个方法会等副线程执行完成之后才会接下来执行主线程,会影响主线程的执行
    thread.join();
    
    System.out.println("我是主线程的执行");
    }
    

    如有违规,请联系博主删除,转载请注明出处!原文链接

  • 相关阅读:
    《分布式系统关注点——数据一致性(上篇)》阅读笔记
    2.23寒假学习记录
    2.22寒假学习记录
    2.21寒假学习记录
    2.20寒假学习记录
    2.19寒假学习记录
    2.18寒假学习记录
    2.17寒假学习记录
    2.17周一毕设改进计划
    2.16寒假学习记录
  • 原文地址:https://www.cnblogs.com/soulmatesjc/p/11213884.html
Copyright © 2020-2023  润新知