java线程基础知识----java daemon线程

时间:2022-03-05 20:05:06

  java线程是一个运用很广泛的重点知识,我们很有必要了解java的daemon线程.

1.首先我们必须清楚的认识到java的线程分为两类: 用户线程和daemon线程

  A.  用户线程: 用户线程可以简单的理解为用户定义的线程,当然包括main线程(以前我错误的认为main线程也是一个daemon线程,但是慢慢的发现原来main线程不是,因为如果我再main线程中创建一个用户线程,并且打出日志,我们会发现这样一个问题,main线程运行结束了,但是我们的线程任然在运行).

  B.  daemon线程: daemon线程是为我们创建的用户线程提供服务的线程,比如说jvm的GC等等,这样的线程有一个非常明显的特征: 当用户线程运行结束的时候,daemon线程将会自动退出.(由此我们可以推出下面关于daemon线程的几条基本特点)

2. daemon 线程的特点:

  A.  守护线程创建的过程中需要先调用setDaemon方法进行设置,然后再启动线程.否则会报出IllegalThreadStateException异常.(个人在想一个问题,为什么不能动态更改线程为daemon线程?有时间一个补上这个内容,现在给出一个猜测: 是因为jvm判断线程状态的时候,如果当前只存在一个线程Thread1,如果我们把这个线程动态更改为daemon线程,jvm会认为当前已经不存在用户线程而退出,稍后将会给出正确结论,抱歉!如果有哪位大牛看到,希望给出指点,谢谢!)

  B.  由于daemon线程的终止条件是当前是否存在用户线程,所以我们不能指派daemon线程来进行一些业务操作,而只能服务用户线程.

  C.  daemon线程创建的子线程任然是daemon线程.

3. 针对上面给出的daemon线程的特点,我们进行如下验证:

  A.  对应上面的A特点:

public class ThreadTest {
public static void main(String[] args) throws InterruptedException {
Thread thread1 = new Thread(new Thread2());
thread1.start();
thread1.setDaemon(true);
Thread.sleep(10);
System.out.println("用户线程退出");
}
}
class Thread2 implements Runnable{
@Override
public void run() {
try {
Thread.sleep(1000);
System.out.println("1+1="+(1+1));
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
运行结果:
Exception in thread "main" java.lang.IllegalThreadStateException
at java.lang.Thread.setDaemon(Thread.java:1352)
at ThreadTest.main(ThreadTest.java:5)
1+1=2

  通过上面的例子我们可以发现,我们并不能动态的更改线程为daemon线程,源码解释如下:

java源码:
public final void setDaemon(boolean on) {
checkAccess();
if (isAlive()) {
throw new IllegalThreadStateException();
}
daemon = on;
}

  我们可以发现,在源码中如果我们的线程状态是alive的,我们的程序就会抛出异常.

  B.对应上面的B特点:

  

public class ThreadTest {
public static void main(String[] args) throws InterruptedException {
Thread thread1 = new Thread(new Thread2());
thread1.setDaemon(true);
thread1.start();
Thread.sleep(10);
System.out.println("用户线程退出");
}
}
class Thread2 implements Runnable{
@Override
public void run() {
try {
Thread.sleep(1000);
System.out.println("1+1="+(1+1));
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
} 运行结果: 用户线程退出

  通过上面的例子我们可以看到,我们在daemon线程中进行相关的计算工作,但是我们并没有获取计算结果,因为用户线程main已经运行结束.

  C.对应上面的C特点:

  

public class ThreadTest {
public static void main(String[] args) throws InterruptedException {
Thread thread = new Thread(new Thread2());
thread.setDaemon(true);
thread.start();
System.out.println("Thread是否时daemon线程"+thread.isDaemon());
Thread.sleep(100);
System.out.println("用户线程退出");
}
}
class Thread2 implements Runnable{
@Override
public void run() {
Thread1 thread1 = new Thread1();
thread1.start();
System.out.println("Thread1是否是daemon线程"+thread1.isDaemon());
}
}
class Thread1 extends Thread{
public void run () {
System.out.println("dosomething");
}
}
运行结果:
Thread是否时daemon线程true
Thread1是否是daemon线程true
dosomething
用户线程退出

  源码解析:

 private void init(ThreadGroup g, Runnable target, String name,
long stackSize, AccessControlContext acc) {
......
this.daemon = parent.isDaemon();
this.priority = parent.getPriority();
......
}

  在进行Thread初始化的时候,会获取父进程的isDaemon来复制子进程的daemon.