多线程之基础篇05:线程生命周期和优先级
标签: 多线程
之前我们简单的介绍过线程的生命周期,但是我们再强调一下
生命周期
说明:
线程共包括以下5种状态:
新建状态(New):
线程对象被创建后,就进入了新建状态。例如,Thread thread = new Thread();
。就绪状态(Runnable):
也被称为“可执行状态”。线程对象被创建后,其它线程调用了该对象的start()方法,从而来启动该线程。例如,thread.start()
。处于就绪状态的线程,随时可能被CPU调度执行。运行状态(Running) :
线程获取CPU权限进行执行。需要注意的是,线程只能从就绪状态进入到运行状态。阻塞状态(Blocked):
阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进入就绪状态,才有机会转到运行状态。阻塞的情况分三种:- 等待阻塞:调用wait(),使该线程处于等待池(wait blocked pool),直到notify()/notifyAll(),线程被唤醒被放到锁定池(lock blocked pool ),释放同步锁使线程回到可运行状态(Runnable)。
- 同步阻塞:对Running状态的线程加同步锁(Synchronized)使其进入(lock blocked pool ),同步锁被释放进入可运行状态(Runnable)。
- 其他阻塞:通过调用线程的sleep()或join()或发出了I/O请求时,线程会进入到阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。
死亡状态(Dead):
线程执行完了或者因异常退出了run()方法,该线程结束生命周期。
优先级
线程的优先级及设置
线程的优先级是为了在多线程环境中便于系统对线程的调度,优先级高的线程将优先执行。
一个线程的优先级设置遵从以下原则:
在一个线程中开启另外一个新线程,则新开线程称为该线程的子线程,子线程初始优先级与父线程相同。
线程创建后,可通过调用setPriority()方法改变优先级。
线程的优先级是1-10之间的正整数。如果什么都没有设置,默认值是5。
1- MIN_PRIORITY
10-MAX_PRIORITY
5-NORM_PRIORITY
注意:
不能依靠线程的优先级来决定线程的执行顺序。与线程休眠类似,线程的优先级仍然无法保障线程的执行次序。只不过,优先级高的线程获取CPU资源的概率较大,优先级低的并非没机会执行。
我们看一下代码演示:
package charThread;
public class Test {
public static void main(String[] args) {
//采用两种不同的方式创建线程
Thread t1 = new MyThread1();
Thread t2 = new Thread(new MyRunnable());
//设置线程优先级
t1.setPriority(10);
t2.setPriority(1);
t2.start();
t1.start();
}
}
class MyThread1 extends Thread {
public void run() {
for (int i = 0; i < 10; i++) {
System.out.println("线程1第" + i + "次执行!");
}
}
}
class MyRunnable implements Runnable {
public void run() {
for (int i = 0; i < 10; i++) {
System.out.println("线程2第" + i + "次执行!");
}
}
}
输出:
线程2第0次执行!
线程1第0次执行!
线程1第1次执行!
线程1第2次执行!
线程1第3次执行!
线程1第4次执行!
线程1第5次执行!
线程2第1次执行!
线程1第6次执行!
线程1第7次执行!
线程1第8次执行!
线程1第9次执行!
线程2第2次执行!
线程2第3次执行!
线程2第4次执行!
线程2第5次执行!
线程2第6次执行!
线程2第7次执行!
线程2第8次执行!
线程2第9次执行!
优先级体现还是蛮明显的。
线程的调度策略
线程调度器选择优先级最高的线程运行。但是,如果发生以下情况,就会终止线程的运行:
线程体中调用了yield()方法,让出了对CPU的占用权。
线程体中调用了sleep()方法,使线程进入睡眠状态,让出CPU资源。
线程由于I/O操作而受阻塞。
在支持时间片的系统中,该线程的时间片用完。