[核心技术36问]17.一个线程两次调用start()方法会出现什么情况?谈谈线程的生命周期和状态转移。

 

d17.一个线程两次调用start()方法会出现什么情况?谈谈线程的生命周期和状态转移。

    Java的线程是不允许启动两次的,第二次调用必然会抛出IllegalThreadStateException,这是一种运行时异常,多次调用start()被认为是编程错误。

    关于线程生命周期的不同状态,在Java 5以后,线程状态被明确定义在java.lang.Thread.State中,分别是:

NEW:表示线程被创建出来还没真正启动的状态,可以认为是一个Java内部状态;

RUNNABLE:表示该线程已经在JVM中执行,当然执行由于执行需要计算资源,它可能是正在运行,也可能还在等待系统分配给它CPU片段,在就绪队列里面排队;

BLOCKED:这个状态和同步非常相关,阻塞表示线程在等待一个Monitor lock。比如,线程试图通过synchronized去获取某个锁,但是其他线程已经独占了,那么当前线程就会处于阻塞状态;

WAITING:表示正在等待其他线程采取某些操作。一个常见的场景是类似生产者消费者模式,发现任务条件尚未满足,就让当前消费者线程等待(wait),另外的生产者线程去准备生产数据,然后通过类似notify等动作,通知消费者线程可以继续工作了。

Thread.join()也会令线程进入等待状态。

TIMED_WAIT:其进入条件和等待状态类似,但是调用的是存在超时条件的方法,比如wait或join等方法的指定超时版本;

TERMINATED:不管是意外退出还是正常执行结束,线程已经完成使命,终止运行,也有人把这个状态叫做死亡。

    在第二次调用start方法的时候,线程可能处于终止或者其他状态(非NEW),但是不论如何,都是不可以再次启动的。

线程是什么?

    线程是系统调度的最小单元,一个进程可以包含多个线程,作为任务的真正调度者,有自己的栈、寄存器、本地存储等,但是会和进程内其他线程共享文件描述符、虚拟地址空间等。

    在具体实现过程中,线程还分为内核线程,用户线程。

    守护线程:有的时候应用中需要一个长期驻留的服务程序,但是不希望其影响应用退出,就可以将其设置为守护线程,如果JVM发现只有守护线程存在时,将结束进程。必须在线程启动之前设置!!

写一个简单的程序,运行这个应用,java至少会创建几个线程?

public class HelloWorld {
	public static void main(String[] args){
		System.out.println("hello world!");
		//int count=Thread.activeCount();
		ThreadGroup group=Thread.currentThread().getThreadGroup();
		ThreadGroup topGroup=group;
		while(group!=null){
			topGroup=group;
			group=group.getParent();
		}
		int nowThreads=topGroup.activeCount();
		Thread[] list=new Thread[nowThreads];
		topGroup.enumerate(list);
		for(int i=0;i<nowThreads;i++){
			System.out.println("线程number:"+i+"="+list[i].getName());
		}
	}

}

  输出:

hello world!
线程number:0=Reference Handler //计算对象是否可达
线程number:1=Finalizer  //回收对象时触发的finalize方法
线程number:2=Signal Dispatcher  //线程调度员
线程number:3=Attach Listener  //在java 6环境中有一个负责接收外部命令的,如jmap,jstack
线程number:4=main

猜你喜欢

转载自blog.csdn.net/hellodake/article/details/81106078