线程的生命周期

线程的生命周期
Java线程具有五种状态:新建、就绪、运行、阻塞、死亡
新建状态(New):当线程对象对创建后,即进入了新建状态,如:Thread t = new MyThread();
就绪状态(Runnable):线程对象对创建后,其他线程调用了该对象的start()方法(t.start();),线程即进入就绪状态。该状态的线程位于可运行线程池中,等待获取CPU的使用权。处于就绪状态的线程,只是说明此线程已经做好了准备,随时等待CPU调度执行,并不是说执行了t.start()此线程立即就会执行;
运行状态(Running):就绪状态的线程获取了CPU,执行程序代码。注:就绪状态是进入到运行状态的唯一入口,也就是说,线程要想进入运行状态执行,首先必须处于就绪状态中;

阻塞状态(Blocked):处于运行状态中的线程由于某种原因,暂时放弃对CPU的使用权,停止执行,此时进入阻塞状态,直到其进入到就绪状态,才有机会再次被CPU调用以进入到运行状态。根据阻塞产生的原因不同,阻塞状态又可以分为三种:
1.等待阻塞:运行状态中的线程执行wait()方法,该线程会释放占用的所有资源,JVM会把该线程放入“等待池”中。进入这个状态后,是不能自动唤醒的,必须依靠其他线程调用notify或notifyAll方法才能唤醒,wait是object类的方法;
2.同步阻塞:行状态中的线程在获取对象的同步锁时,若该同步锁被其它线程所占用,则JVM会把该线程放入“锁池”中
3.其他阻塞:运行状态中的线程执行sleep()或join()方法,或发出了I/O请求时,线程会进入到阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。sleep是Thread类的方法。

死亡状态(Dead):线程执行完了或者因异常退出了run()方法,该线程结束生命周期。

sleep()、wait()、join()、yield()的区别
1.锁池
所有需要竞争同步锁的线程都会放在锁池中,比如当前对象的锁已经被其中一个线程得到,则其他线程需要这个锁池进行等待,当前面的线程释放同步锁后,锁池中的线程去竞争同步锁,当某个线程得到后就会进入就绪队列,等待cpu资源分配。
2.等待池
当我们调用wait()方法后,线程会被放到等待池中,等待池的线程不会去竞争同步锁,只有调用了notify()或notifyAll()后等待池的线程才会开始去竞争锁,notify()是随机从等待池选出一个线程放到锁池,notifyAll()是将等待池的所有线程放到锁池中

1、sleep是Thread类的静态本地方法,wait则是Object类的本地方法。
2、sleep方法不会释放锁,但是wait会释放,而且线程会被放到等待队列中。
sleep就是把cpu的执行资格和执行权释放出去,不再运行此线程,当定时时间结束后再取回cpu资源,参与cpu的调度,获取到cpu资源后就可以继续运行了。而如果sleep时该线程有锁,那么sleep不会释放这个锁,而是把锁带着进入了冻结状态,也就是说其他需要这个锁的线程根本不可能获取到这个锁。也是就是说无法执行程序。如果在睡眠期间其他线程调用了这个线程的interrupt方法,那么这个线程也会抛出interruptException异常返回,这点和wait是一样的。
3、sleep方法不依赖于同步器synchronized,但是wait需要依赖synchronized关键字。
4、sleep不需要被唤醒(休眠之后退出阻塞),但是wait需要(不指定时间时需要被唤醒)
5、sleep一般用于当前线程休眠,或者轮循暂停操作,wait则多用于多线程之间的通信。
6、sleep会让出cpu执行时间且强制cpu上下文切换,而wait则不一定,wait后可能还是有机会重新竞争到锁继续执行的。
yield()执行后线程进入就绪状态,马上释放了cpu的执行权,但是依然保留了cpu的执行资格(就绪状态),所以有可能cpu下次进行线程调度还会让这个线程获取到执行权继续执行。
join()执行后线程进入阻塞状态,例如在线程b中调用线程a的join(),那么线程b就会进入到阻塞队列,直到线程a结束或中断线程

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值