Java中的父线程与子线程

news/2024/11/9 0:54:01/

转自: https://my.oschina.net/hosee/blog/509557

以前在学习操作系统的时候,一直记得的父线程死后,子线程也消失了。然而今天在查资料中,发现有点疑惑,在此记录一下。

Java编写的程序都运行在Java虚拟机(JVM)中,在JVM的内部,程序的多任务是通过线程来实现的。

每用java命令启动一个java应用程序,就会启动一个JVM进程。在同一个JVM进程中,有且只有一个进程,就是它自己。在这个JVM环境中,所有程序代码的运行都是以线程来运行的。JVM找到程序的入口点main(),然后运行main()方法,这样就产生了一个线程,这个线程称之为主线程。当main方法结束后(没有其他线程时),主线程运行完成。JVM进程也随即退出。

操作系统将进程线程进行管理,轮流(没有固定的顺序)分配每个进程很短的一段时间(不一定是均分),然后在每个进程内部,程序代码自己处理该进程内部线程的时间分配,多个线程之间相互的切换去执行,这个切换时间也是非常短的。

对于程序来说,如果主进程在子进程还未结束时就已经退出,那么Linux内核会将子进程的父进程ID改为1(也就是init进程),当子进程结束后会由init进程来回收该子进程。

那如果是把进程换成线程的话,会怎么样呢?假设主线程在子线程结束前就已经退出,子线程会发生什么?

首先我们来看一个网上很多人的例子:

 

package test;public class Test1 extends Thread
{@Overridepublic void run(){while (true){try{Thread.sleep(2000);}catch (InterruptedException e){// TODO Auto-generated catch blocke.printStackTrace();}System.out.println("我还活着");}}public static void main(String[] args) throws InterruptedException{Thread t = new Test1();t.start();Thread.sleep(5000);System.out.println("Main End");}
}

 

 

输出:

 

 

我还活着
我还活着
Main End
我还活着
我还活着

 

 

上文说了 JVM找到程序的入口点main(),然后运行main()方法,这样就产生了一个线程,这个线程称之为主线程。当main方法结束后(没有其他线程时),主线程运行完成。JVM进程也随即退出。然而上述输出表明当main()运行到最后后,子线程依然在输出。所以大家就得出了结论,父线程要等待子线程完成后才会退出。然而我们再看个例子:

 

 

package test;public class Test extends Thread
{@Overridepublic void run(){Thread sonthread = new a();sonthread.start();}public static void main(String[] args) throws InterruptedException{Thread fatherThread = new Test();fatherThread.start();Thread.sleep(5000);fatherThread.interrupt();Thread.sleep(2000);System.out.println("fatherThread.isAlive()?  "+fatherThread.isAlive());}
}class a extends Thread
{@Overridepublic void run(){while (true){try{Thread.sleep(1000);}catch (InterruptedException e){e.printStackTrace();}System.out.println("我还活着");}}
}

 

 

输出:

 

 

我还活着
我还活着
我还活着
我还活着
我还活着
我还活着
fatherThread.isAlive()?  false
我还活着
我还活着
我还活着

 

 

很明显,父线程死后子线程还在输出。两个例子到底哪个是正确的呢?

 

查了很多资料得到了解答。

如果main方法中没有创建其他线程,那么当main方法返回时JVM就会结束Java应用程序。但如果main方法中创建了其他线程,那么JVM就要在主线程和其他线程之间轮流切换,保证每个线程都有机会使用CPU资源,main方法返回(主线程结束)JVM也不会结束,要一直等到该程序所有线程全部结束才结束Java程序(另外一种情况是:程序中调用了Runtime类的exit方法,并且安全管理器允许退出操作发生。这时JVM也会结束该程序)。

那么又有个思考,JVM是怎么知道线程都结束的呢?

JVM中有一个线程DestroyJavaVM,执行main()的线程在main执行完后调用JNI中的jni_DestroyJavaVM()方法唤起DestroyJavaVM线程。JVM在Jboss服务器启动之后,就会唤起DestroyJavaVM线程,处于等待状态,等待其它线程(java线程和native线程)退出时通知它卸载JVM。线程退出时,都会判断自己当前是否是整个JVM中最后一个非deamon线程,如果是,则通知DestroyJavaVM线程卸载JVM。ps:扩展一下:1.如果线程退出时判断自己不为最后一个非deamon线程,那么调用thread->exit(false),并在其中抛出thread_end事件,jvm不退出。2.如果线程退出时判断自己为最后一个非deamon线程,那么调用before_exit()方法,抛出两个事件: 事件1:thread_end线程结束事件、事件2:VM的death事件。然后调用thread->exit(true)方法,接下来把线程从active list卸下,删除线程等等一系列工作执行完成后,则通知正在等待的DestroyJavaVM线程执行卸载JVM操作。

所以第一个例子时,主线程运行完,但是它不是最后一个非守护线程,所以JVM并没有退出,所以子线程还会继续运行。

第二个例子。主线程一直在,所以JVM不会退出。当父线程死去后,子线程还在运行。说明父线程的生命周期与子线程没有关系。

参考资料:

1. http://warnerhit.iteye.com/blog/1407484

2. http://liyuanlife.com/blog/2015/04/08/influence-of-main-threads-exiting-to-child-thread/

3. http://blog.yemou.net/article/query/info/tytfjhfascvhzxcytp47

4. http://jinguo.iteye.com/blog/747256


http://www.ppmy.cn/news/662169.html

相关文章

JAVA锁的膨胀过程

为什么80%的码农都做不了架构师?>>> 首先简单说下先偏向锁、轻量级锁、重量级锁三者各自的应用场景: 偏向锁:只有一个线程进入临界区;轻量级锁:多个线程交替进入临界区;重量级锁:多…

ConcurrentHashMap 分段锁

ConcurrentHashMap 继承自AbstractMap类&#xff08;AbstractMap实现了Map接口&#xff09;&#xff0c;用来替换其他线程安全的Map容器。例如&#xff1a; Hashtable 和Collections.synchronizedMap(new HashMap<String,String>())&#xff1b; 另外&#xff0c;说到Co…

behavios

什么是behavios&#xff1f; behaviors 是小程序中&#xff0c;用于实现组件间代码共享的特性&#xff0c;类似于 Vue.js 中的 “mixins”。 behaviors 的工作方式&#xff1a; 每个 behavior 可以包含一组属性、数据、生命周期函数和方法。组件引用它时&#xff0c;它的属性…

Shapes

系列文章 SFML-windows 篇 SFML-Events explained 篇 SFML-Keyboard, mouse and joystick 篇 SFML-Using OpenGL in a SFML window 篇 SFML-Drawing 2D stuff 篇 SFML-Shapes 篇 SFML-Sprites and textures 篇 文章目录 系列文章一、Introduction二、Common shape properties颜…

ConcurrentHashMap 总结( 上 )

来源&#xff1a;Hosee&#xff0c; my.oschina.net/hosee/blog/675884 并发编程实践中&#xff0c;ConcurrentHashMap是一个经常被使用的数据结构&#xff0c;相比于Hashtable以及Collections.synchronizedMap()&#xff0c;ConcurrentHashMap在线程安全的基础上提供了更好的写…

跨域小结(为什么form表单提交没有跨域问题,但ajax提交有跨域问题)

原文链接&#xff1a;https://www.zhihu.com/question/31592553 ------------------------------------- ------------------------------------------- ------------------------------------ 跨域springmvc http://blog.csdn.net/wabiaozia/article/details/52778335 跨域…

预测算法-三次指数平滑法(Holt-Winters)

指数平滑 一次指数平滑 一次指数平滑法是一种特殊的加权平均法&#xff0c;对本期观察值和本期预测值赋予不同的权重&#xff0c;求得下一期预测值的方法。这种方法既不需要存储全部历史数据&#xff0c;也不需要存储一组数据&#xff0c;从而可以大大减少数据存储问题。其通…

Search

Search 简介 什么时Search Transwarp Search基于开源的Elasticsearch进行开发&#xff0c;在其架构的基础上进行了多种功能优化&#xff0c;并通过Esdrive实现了SQL交互方式。Transwarp Search是一个可扩展的分布式全文搜索和分析引擎。 在TDH中Search主要扮演两个角色。首先…