线程有自己的状态,看下图:
上图基本描述的线程的生命状态,有的地方也会将冻结状态分为两种,但这没多大意义,明白就好,上图的流程也有少画的,详细会解释。解释一下各种状态
消亡状态:不用说,线程死了。
创建状态:new Thread()就代表创建了一个线程
运行状态:thread.start()之后就开始运行现场,不过有可用没有抢到CPU执行权,所以有可能在阻塞状态等待。
冻结状态:sleep或者wait会引起冻结状态,何为冻结状态,就是线程主动放弃了执行资格(sleep和wait的区别下面会做解释)。
阻塞状态:线程有执行资格,但没有执行权,何为执行权?就是CPU没轮到它呢。
sleep与wait的区别:
sleep(毫秒数) : 使用这种方式放弃的执行资格,不会放弃它持有的锁。
wait:使用这种方式放弃的执行资格,会放弃它持有的锁,并且进到锁池,等待其他持有此对象的线程调用了notify或者notifyAll方法它才能重新拿到锁并且回到阻塞状态等待执行。
notify或者notifyAll:
notify:只会唤醒一个线程,并且是谁先wait的先唤醒谁。
notifyAll:会唤醒此对象锁的锁池中所有的线程。
wait和notify,notifyAll方法容易产生的错误:
这三个方法的调用必须是在有锁的状态下调用,否则会抛出监控器状态异常的异常。
还有为什么这三个方法要定义在Object中?
实例一段代码:
public void run() { synchronized(this){ try { this.wait(); } catch (InterruptedException e) { e.PRintStackTrace(); } this.notify(); } }这三个方法的调用,默认会在前面加上锁对象的引用,而锁对象可以是任意对象,所以这三个方法应该可以被任意对象调用,被任意对象可以调用的方法必定定义在Object中。notify和notifyAll方法的注意点:
notify和notifyAll唤醒的都是它的对象锁的锁池中的线程。注意锁不一样,是无法唤醒对方的锁池中的线程的。
新闻热点
疑难解答