启动一个线程的方法是_为什么start方法才能启动线程,而run不行?

我们都知道,一个线程直接对应了一个Thread对象,在刚开始学习线程的时候我们也知道启动线程是通过start()方法,而并非run()方法。

那这是为什么呢?

如果你熟悉Thread的代码的话,你应该知道在这个类加载的时候会注册一些native方法

9e59275b63d0cb2d17b7c4a4416f4dfd.png

一看到native我就想起了JNI,registerNatives()实际上就是java方法和C/C++的函数对应。在首次加载的时候就会注册这些native方法。Thread中有很多native方法,大家有兴趣的可以去看看。

关于JNI方法的命名,我们可以这样测试,我们用java声明一个native方法,然后先使用javac编译源文件(比如javac main.java),然后在使用javah即可生成头文件(javah main),打开这个头文件你就知道方法命名是如何的了

我们在JVM源码中搜索Java_java_lang_Thread_registerNatives可以看到registerNatives方法的具体实现

138e33fecb57414a21648a57c1c297e7.png

可以看到,在registerNatives函数中,注册了很多的native方法比如这里的start0()方法。

所有对JNI函数的调用都使用了env指针,该指针是对每一个本地方法的第一个参数。env指针是函数指针表的指针。我们可以在docs.oracle.com/javase/8/do…中找到JNI API

在Thread.start()方法中,实际就是通过调用start0()方法来启动线程的。

4bd422e46abe1ba71bad047f72e85c75.png

而JNINativeMethod这个数据结构定义如下:

b3a9aac7a5a6dfcb1ba01a91d267d8b4.png

因此start0()这个方法对应的本地函数是JVM_StartThread

571426a34517f6053391f0dac95565a4.png

我们接下来看JVM_StartThread的方法实现

df107a84827da14fe7038d3515558c0b.png

上面代码主要做了三件事情

  1. 判断当前线程状态是否合法,不合法抛出IllegalThreadStateException
  2. 创建一个Java线程(我们需要重点关注的)
  3. 将线程状态设置为Runnable

如果面试官以后再问你两次调用start()方法会怎样,你就大胆而坚定的回复说抛出IllegalThreadStateException。

在JavaThread构造函数中实际调用的是os::create_thread方法

0264ef1d2796723079f010ef8ec05493.png

pthread_create函数作用是创建一个线程,它的第三个参数是线程运行函数的起始地址,第四个参数是运行函数参数

IEEE标准1003.1c中定义了线程的标准,它定义的线程包叫做Pthread,大部分UNIX系统都支持这个标准。

我们的thread_native_entry实际传入的是JavaThread这个对象,所以最终会调用JavaThread::run()(thread.cpp中)

aef705944f512de9ac168d5fa80cf3f0.png

thread_main_inner函数中entry_point的返回值实际上是我们在创建JavaThread的时候传入的第一个参数thread_entry。而thread_entry指针指向的函数如下:

5caca73380d883f2a6c81a6c117d2e45.png

这样我们就最终通过JavaCalls调用了run方法。

总结

new Thread只是创建了一个普通的Java对象,只有在调用了start()方法之后才会创建一个真正的线程,在JVM内部会在创建线程之后调用run()方法,执行相应的业务逻辑。

由于Java中线程最终还是和操作系统线程挂钩了的,所以线程资源是一个很重要的资源,为了复用我们一般是通过线程池的方式来使用线程。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值