守护线程与join
守护线程
所谓守护线程是指在程序运行的时候在后台提供一种通用服务的线程,比如垃圾回收线程就是一个很称职的守护者,并且这种线程并不属于程序中不可或缺的部分。因 此,当所有的非守护线程结束时,程序也就终止了,同时会杀死进程中的所有守护线程。反过来说,只要任何非守护线程还在运行,程序就不会终止。
守护线程和用户线程的没啥本质的区别:唯一的不同之处就在于虚拟机的离开:如果用户线程已经全部退出运行了,只剩下守护线程存在了,虚拟机也就退出了。 因为没有了被守护者,守护线程也就没有工作可做了,也就没有继续运行程序的必要了。
注意:
(1) thread.setDaemon(true)必须在thread.start()之前设置,否则会跑出一个IllegalThreadStateException异常。也就是不能把正在运行的常规线程设置为守护线程。
(2)在Deamon线程中产生的新线程也是Daemon的
(3)守护线程应该永远不去访问固有资源,如文件
数据库,因为它会在任何时候甚至在一个操作的中间发生中断,我们无法预期的。
public class Test1 {
public static void main(String[] args) throws Exception {
Thread thread = new Thread(() -> {
try {
System.out.println(Thread.currentThread().getName());
TimeUnit.SECONDS.sleep(10);
System.out.println(Thread.currentThread().getName());
} catch (InterruptedException e) {
}
});
thread.setDaemon(true);
thread.start();
/**
public final void setDaemon(boolean on) {
checkAccess();
//如果线程已经运行了,再设置其为守护线程会异常
if (isAlive()) {
throw new IllegalThreadStateException();
}
daemon = on;
}
*/
TimeUnit.SECONDS.sleep(2);
System.out.println(Thread.currentThread().getName());
}
}
join
t.join()方法阻塞调用此方法的线程(call thread),直到线程t完成,此线程(call thread)再继续;通常用于在main()主线程内,等待其他线程完成再结束main()主线程。
public class Test3 {
public static void main(String[] args) {
Thread t1 = new Thread(() -> {
IntStream.range(1,1000).forEach(i -> System.out.println(Thread.currentThread().getName() + " -> " + i));
});
t1.start();
Thread t2 = new Thread(() -> {
IntStream.range(1,1000).forEach(i -> System.out.println(Thread.currentThread().getName() + " -> " + i));
});
t2.start();
try {
// join相对于的当前线程,并不不是指t1.join()在t2.join()
// 前面就会出现t2在t1结束后才运行
t1.join();
t2.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
IntStream.range(1,1000).forEach(i -> System.out.println(Thread.currentThread().getName() + " -> " + i));
}
}