多线程学习(二)——线程的生命周期

14 篇文章 0 订阅
11 篇文章 0 订阅

        线程被创建并启动后,它既不是一启动就进入执行状态,也不是一直处于执行状态,线程的生命周期包括:

        新建(New)、就绪(Runnable)、运行(Running)、阻塞(Blocked)和死亡(Dead)

       五种状态。线程启动以后,它不可能一直独自占着CPU,CPU需要在多条线程之间切换,所以线程的状态也会多次在运行,阻塞之间切换。

新建和就绪状态

       当使用new关键字创建线程后,该线程就处于新建状态,此时他由java虚拟机为其分配内存,并初始化其成员变量的值,这时,程序并不会执行线程的线程执行体。

        当线程对象调用了start()方法后,线程就会处于就绪状态,java虚拟机为其创建方法调用栈和程序计数器,处于这个状态中的程序并没有运行,只是表示该线程可以运行了。线程何时开始运行,取决于JVM里线程调度器的调度。

注意:启动线程使用start()方法,而不是run()方法!永远不要调用线程的run()方法!因为,调用start()方法,启动线程的时候,系统会把run()方法当成线程的执行体来处理;但是如果直接调用线程对象的run()方法,则run()方法会被立即执行,而且系统会把线程对象当成一个普通的对象,把run() 方法看成一个普通方法,而不是线程执行体。

注意:只能对处于新建状态的线程调用start()方法,否则将引发IllegalThreadStateException异常。

运行和阻塞状态

       线程运行后,不可能一直处于运行状态(除非它的线程体足够的短,瞬间就执行结束了),线程在运行中需要被中断,目的是让其他线程获得执行的机会,线程调度的细节取决于底层平台所采用的策略。现代的桌面和服务操作系统都采用了抢占式调度策略(系统会给每个可执行的线程一个小时间段来处理任务,当这个时间段用完后,系统会剥夺该线程所占用的资源,让其他线程获得执行的机会),但是一些小型设备如手机可能采取的是协作式调度策略,在这样的系统中,只有当一个线程调用了它的sleep()或者yield()方法后,才会放弃所占用的系统资源——也就是必须由该线程主动放弃所占用资源。

                

                                                                      线程状态转换图

      上图可以看到,线程从阻塞状态只能进入就绪状态,无法直接进入运行状态。而就绪状态和运行状态间的转换通常不受程序的控制,而是由系统线程调度所决定的。

线程死亡

线程会以下面三种方式结束,结束后就处于死亡状态:

1、run()或者call()方法执行完成,线程正常结束

2、线程抛出一个未捕获的Exception或Error

3、直接调用该线程的stop()方法来结束该线程——这个方法容易导致死锁,通常不推荐使用

注意:当主线程结束时,其他线程并不受任何影响,并不会随之结束。一旦子线程启动起来后,它就具有和主线程一样的地位,不会受主线程的影响。

注意:不要试图对一个已死亡的线程调用start()方法使他重新启动,死亡就是死亡,这个线程不可再次作为线程执行。

下面尝试对处于死亡状态的线程再次调用start()方法:

public class StartDead extends Thread{
	
	private int i;
	//重写run()方法,run()方法的方法执行体就是线程执行体
	public void run() {
		for(;i<100;i++) {
			System.out.println(currentThread().getName()+" "+i);
		}
	}
	public static void main(String[] args) {
		//创建线程对象
		StartDead sd = new StartDead();
		for(int i = 0;i<100;i++) {
			System.out.println(Thread.currentThread().getName() + " " + i);
			if(i==20) {
				//启动线程
				sd.start();
				//判断启动后线程的isAlive()方法返回true
				System.out.println(sd.isAlive());
			}
			//当线程处于新建、死亡两种状态时,isAlive()方法返回false
			//当i>20时,该线程肯定已经启动过了,如果sd.isAlive()为假时
			//那就是死亡状态了
			if(i>20&&!sd.isAlive()) {
				//试图再次启动该线程
				sd.start();
			}
		}
	}

}

运行结果如下(截取部分结果):



从上面结果,可以看到对已死亡的线程调用start()方法来再次启动线程时,会引发IllegalThreadStateException异常,这表明处于死亡状态的线程无法再次运行了

注意

1、不要对处于死亡状态的线程调用start()方法,程序只能对新建状态的线程调用start()方法

2、对新建状态的线程两次调用start()方法也是错误的

上面两种均会引发IllegalThreadStateException异常。

(参考《疯狂Java讲义第3版》)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值