Java線程的start方法回調(diào)run方法的操作技巧
面試中可能會(huì)被問到為什么我們調(diào)用start()方法時(shí)會(huì)執(zhí)行run()方法,為什么我們不能直接調(diào)用run()方法?
Java 創(chuàng)建線程的方法
實(shí)際上,創(chuàng)建線程最重要的是提供線程函數(shù)(回調(diào)函數(shù)),該函數(shù)作為新創(chuàng)建線程的入口函數(shù),實(shí)現(xiàn)自己想要的功能。Java 提供了兩種方法來創(chuàng)建一個(gè)線程:
繼承 Thread 類
class MyThread extends Thread{
public void run() {
System.out.println("My thread is started.");
}
}
實(shí)現(xiàn)該繼承類的 run 方法,然后就可以創(chuàng)建這個(gè)子類的對(duì)象,調(diào)用 start 方法即可創(chuàng)建一個(gè)新的線程:
MyThread myThread = new MyThread(); myThread.start();
實(shí)現(xiàn) Runnable 接口
class MyRunnable implements Runnable{
public void run() {
System.out.println("My runnable is invoked.");
}
}
實(shí)現(xiàn) Runnable 接口的類的對(duì)象可以作為一個(gè)參數(shù)傳遞到創(chuàng)建的 Thread 對(duì)象中,同樣調(diào)用 Thread#start 方法就可以在一個(gè)新的線程中運(yùn)行 run 方法中的代碼了。
Thread myThread = new Thread( new MyRunnable()); myThread.start();
可以看到,不管是用哪種方法,實(shí)際上都是要實(shí)現(xiàn)一個(gè) run 方法的。 該方法本質(zhì)是上一個(gè)回調(diào)方法。由 start 方法新創(chuàng)建的線程會(huì)調(diào)用這個(gè)方法從而執(zhí)行需要的代碼。 從后面可以看到,run 方法并不是真正的線程函數(shù),只是被線程函數(shù)調(diào)用的一個(gè) Java 方法而已,和其他的 Java 方法沒有什么本質(zhì)的不同。
Java 線程的實(shí)現(xiàn)
從概念上來說,一個(gè) Java 線程的創(chuàng)建根本上就對(duì)應(yīng)了一個(gè)本地線程(native thread)的創(chuàng)建,兩者是一一對(duì)應(yīng)的。 問題是,本地線程執(zhí)行的應(yīng)該是本地代碼,而 Java 線程提供的線程函數(shù)是 Java 方法,編譯出的是 Java 字節(jié)碼,所以可以想象的是, Java 線程其實(shí)提供了一個(gè)統(tǒng)一的線程函數(shù),該線程函數(shù)通過 Java 虛擬機(jī)調(diào)用 Java 線程方法 , 這是通過 Java 本地方法調(diào)用來實(shí)現(xiàn)的。
以下是 Thread#start 方法的示例:
public synchronized void start() {
…
start0();
…
}
可以看到它實(shí)際上調(diào)用了本地方法 start0, 該方法的聲明如下:
private native void start0();
Thread 類有個(gè) registerNatives 本地方法,該方法主要的作用就是注冊(cè)一些本地方法供 Thread 類使用,如 start0(),stop0() 等等,可以說,所有操作本地線程的本地方法都是由它注冊(cè)的 . 這個(gè)方法放在一個(gè) static 語句塊中,這就表明,當(dāng)該類被加載到 JVM 中的時(shí)候,它就會(huì)被調(diào)用,進(jìn)而注冊(cè)相應(yīng)的本地方法。
private static native void registerNatives();
static{
registerNatives();
}
本地方法 registerNatives 是定義在 Thread.c 文件中的。Thread.c 是個(gè)很小的文件,定義了各個(gè)操作系統(tǒng)平臺(tái)都要用到的關(guān)于線程的公用數(shù)據(jù)和操作,如代碼清單 1 所示。
清單1
JNIEXPORT void JNICALL
Java_Java_lang_Thread_registerNatives (JNIEnv *env, jclass cls){
(*env)->RegisterNatives(env, cls, methods, ARRAY_LENGTH(methods));
}
static JNINativeMethod methods[] = {
{"start0", "()V",(void *)&JVM_StartThread},
{"stop0", "(" OBJ ")V", (void *)&JVM_StopThread},
{"isAlive","()Z",(void *)&JVM_IsThreadAlive},
{"suspend0","()V",(void *)&JVM_SuspendThread},
{"resume0","()V",(void *)&JVM_ResumeThread},
{"setPriority0","(I)V",(void *)&JVM_SetThreadPriority},
{"yield", "()V",(void *)&JVM_Yield},
{"sleep","(J)V",(void *)&JVM_Sleep},
{"currentThread","()" THD,(void *)&JVM_CurrentThread},
{"countStackFrames","()I",(void *)&JVM_CountStackFrames},
{"interrupt0","()V",(void *)&JVM_Interrupt},
{"isInterrupted","(Z)Z",(void *)&JVM_IsInterrupted},
{"holdsLock","(" OBJ ")Z",(void *)&JVM_HoldsLock},
{"getThreads","()[" THD,(void *)&JVM_GetAllThreads},
{"dumpThreads","([" THD ")[[" STE, (void *)&JVM_DumpThreads},
};
到此,可以容易的看出 Java 線程調(diào)用 start 的方法,實(shí)際上會(huì)調(diào)用到 JVM_StartThread 方法,那這個(gè)方法又是怎樣的邏輯呢。實(shí)際上,我們需要的是(或者說 Java 表現(xiàn)行為)該方法最終要調(diào)用 Java 線程的 run 方法,事實(shí)的確如此。 在 jvm.cpp 中,有如下代碼段:
JVM_ENTRY(void, JVM_StartThread(JNIEnv* env, jobject jthread)) … native_thread = new JavaThread(&thread_entry, sz);
**這里JVM_ENTRY是一個(gè)宏,用來定義**JVM_StartThread 函數(shù),可以看到函數(shù)內(nèi)創(chuàng)建了真正的平臺(tái)相關(guān)的本地線程,其線程函數(shù)是 thread_entry,如清單 2 所示。
清單2
static void thread_entry(JavaThread* thread, TRAPS) {
HandleMark hm(THREAD);
Handle obj(THREAD, thread->threadObj());
JavaValue result(T_VOID);
JavaCalls::call_virtual(&result,obj,
KlassHandle(THREAD,SystemDictionary::Thread_klass()),
vmSymbolHandles::run_method_name(),
vmSymbolHandles::void_method_signature(),THREAD);
}
可以看到調(diào)用了 vmSymbolHandles::run_method_name 方法,這是在 vmSymbols.hpp 用宏定義的:
class vmSymbolHandles: AllStatic {
…
template(run_method_name,"run")
…
}
至于 run_method_name 是如何聲明定義的,因?yàn)樯婕暗胶芊爆嵉拇a細(xì)節(jié),本文不做贅述。感興趣的讀者可以自行查看 JVM 的源代碼。
圖. Java 線程創(chuàng)建調(diào)用關(guān)系圖

start() 創(chuàng)建新進(jìn)程
run() 沒有
PS:下面看下Java線程中run和start方法的區(qū)別
Thread類中run()和start()方法的區(qū)別如下:
run()方法:在本線程內(nèi)調(diào)用該Runnable對(duì)象的run()方法,可以重復(fù)多次調(diào)用;
start()方法:啟動(dòng)一個(gè)線程,調(diào)用該Runnable對(duì)象的run()方法,不能多次啟動(dòng)一個(gè)線程;
package com.ljq.test;
public class ThreadTest {
/**
* 觀察直接調(diào)用run()和用start()啟動(dòng)一個(gè)線程的差別
*
* @param args
* @throws Exception
*/
public static void main(String[] args){
Thread thread=new ThreadDemo();
//第一種
//表明: run()和其他方法的調(diào)用沒任何不同,main方法按順序執(zhí)行了它,并打印出最后一句
//thread.run();
//第二種
//表明: start()方法重新創(chuàng)建了一個(gè)線程,在main方法執(zhí)行結(jié)束后,由于start()方法創(chuàng)建的線程沒有運(yùn)行結(jié)束,
//因此主線程未能退出,直到線程thread也執(zhí)行完畢.這里要注意,默認(rèn)創(chuàng)建的線程是用戶線程(非守護(hù)線程)
//thread.start();
//第三種
//1、為什么沒有打印出100句呢?因?yàn)槲覀儗hread線程設(shè)置為了daemon(守護(hù))線程,程序中只有守護(hù)線程存在的時(shí)候,是可以退出的,所以只打印了七句便退出了
//2、當(dāng)java虛擬機(jī)中有守護(hù)線程在運(yùn)行的時(shí)候,java虛擬機(jī)會(huì)關(guān)閉。當(dāng)所有常規(guī)線程運(yùn)行完畢以后,
//守護(hù)線程不管運(yùn)行到哪里,虛擬機(jī)都會(huì)退出運(yùn)行。所以你的守護(hù)線程最好不要寫一些會(huì)影響程序的業(yè)務(wù)邏輯。否則無法預(yù)料程序到底會(huì)出現(xiàn)什么問題
//thread.setDaemon(true);
//thread.start();
//第四種
//用戶線程可以被System.exit(0)強(qiáng)制kill掉,所以也只打印出七句
thread.start();
System.out.println("main thread is over");
System.exit(1);
}
public static class ThreadDemo extends Thread{
@Override
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println("This is a Thread test"+i);
}
}
}
}
總結(jié)
以上所述是小編給大家介紹的Java線程的start方法回調(diào)run方法的操作技巧,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
Spring動(dòng)態(tài)配置計(jì)時(shí)器觸發(fā)時(shí)間的實(shí)例代碼
這篇文章主要介紹了Spring動(dòng)態(tài)配置計(jì)時(shí)器觸發(fā)時(shí)間的實(shí)例代碼,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-06-06
SpringBoot 統(tǒng)一異常處理的實(shí)現(xiàn)示例
本文主要介紹了SpringBoot 統(tǒng)一異常處理的實(shí)現(xiàn)示例,目的就是在異常發(fā)生時(shí),盡可能地減少破壞,下面就來介紹一下,感興趣的可以了解一下2024-07-07
Mybatis中and和循環(huán)or混用操作(or轉(zhuǎn)換成in)
這篇文章主要介紹了Mybatis中and和循環(huán)or混用操作(or轉(zhuǎn)換成in),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
SpringSecurity6.x多種登錄方式配置小結(jié)
SpringSecurity6.x變了很多寫法,本文就來介紹一下SpringSecurity6.x多種登錄方式配置小結(jié),具有一定的參考價(jià)值,感興趣的可以了解一下2023-12-12
javascript與jsp發(fā)送請(qǐng)求到servlet的幾種方式實(shí)例
本文分別給出了javascript發(fā)送請(qǐng)求到servlet的5種方式實(shí)例與 jsp發(fā)送請(qǐng)求到servlet的6種方式實(shí)例2018-03-03
Spring MVC多種情況下進(jìn)行文件上傳的實(shí)例
上傳是Web工程中很常見的功能,SpringMVC框架簡(jiǎn)化了文件上傳的代碼,本文給大家總結(jié)了Spring MVC多種情況下進(jìn)行文件上傳的實(shí)例,并通過代碼示例給大家介紹的非常詳細(xì),需要的朋友可以參考下2024-02-02
詳解Spring Cloud Feign 熔斷配置的一些小坑
這篇文章主要介紹了詳解Spring Cloud Feign 熔斷配置的一些小坑,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-04-04
java反射實(shí)現(xiàn)javabean轉(zhuǎn)json實(shí)例代碼
基于java反射機(jī)制實(shí)現(xiàn)javabean轉(zhuǎn)json字符串實(shí)例,大家參考使用吧2013-12-12
Java 在PPT中添加文本和圖片超鏈接的實(shí)現(xiàn)方法
這篇文章主要介紹了Java 在PPT中添加文本和圖片超鏈接的實(shí)現(xiàn)方法,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-05-05

