亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java線程的start方法回調(diào)run方法的操作技巧

 更新時間:2017年11月10日 09:53:39   作者:麥田  
面試過程中經(jīng)常會被面試官問到為什么我們調(diào)用start()方法時會執(zhí)行run()方法,為什么不能直接調(diào)用run()方法,問的一頭霧水,今天小編給大家介紹下Java線程的start方法回調(diào)run方法的操作技巧,需要的朋友參考下吧

面試中可能會被問到為什么我們調(diào)用start()方法時會執(zhí)行run()方法,為什么我們不能直接調(diào)用run()方法?

Java 創(chuàng)建線程的方法

實際上,創(chuàng)建線程最重要的是提供線程函數(shù)(回調(diào)函數(shù)),該函數(shù)作為新創(chuàng)建線程的入口函數(shù),實現(xiàn)自己想要的功能。Java 提供了兩種方法來創(chuàng)建一個線程:

繼承 Thread 類

class MyThread extends Thread{ 
 public void run() { 
  System.out.println("My thread is started."); 
 } 
}

實現(xiàn)該繼承類的 run 方法,然后就可以創(chuàng)建這個子類的對象,調(diào)用 start 方法即可創(chuàng)建一個新的線程:

MyThread myThread = new MyThread(); 
myThread.start();

實現(xiàn) Runnable 接口

class MyRunnable implements Runnable{ 
 public void run() { 
  System.out.println("My runnable is invoked."); 
 } 
}

實現(xiàn) Runnable 接口的類的對象可以作為一個參數(shù)傳遞到創(chuàng)建的 Thread 對象中,同樣調(diào)用 Thread#start 方法就可以在一個新的線程中運行 run 方法中的代碼了。

Thread myThread = new Thread( new MyRunnable()); 
myThread.start();

可以看到,不管是用哪種方法,實際上都是要實現(xiàn)一個 run 方法的。 該方法本質(zhì)是上一個回調(diào)方法。由 start 方法新創(chuàng)建的線程會調(diào)用這個方法從而執(zhí)行需要的代碼。 從后面可以看到,run 方法并不是真正的線程函數(shù),只是被線程函數(shù)調(diào)用的一個 Java 方法而已,和其他的 Java 方法沒有什么本質(zhì)的不同。

Java 線程的實現(xiàn)

從概念上來說,一個 Java 線程的創(chuàng)建根本上就對應了一個本地線程(native thread)的創(chuàng)建,兩者是一一對應的。 問題是,本地線程執(zhí)行的應該是本地代碼,而 Java 線程提供的線程函數(shù)是 Java 方法,編譯出的是 Java 字節(jié)碼,所以可以想象的是, Java 線程其實提供了一個統(tǒng)一的線程函數(shù),該線程函數(shù)通過 Java 虛擬機調(diào)用 Java 線程方法 , 這是通過 Java 本地方法調(diào)用來實現(xiàn)的。

以下是 Thread#start 方法的示例:

public synchronized void start() { 
 …
 start0(); 
 …
}

可以看到它實際上調(diào)用了本地方法 start0, 該方法的聲明如下:

private native void start0();

Thread 類有個 registerNatives 本地方法,該方法主要的作用就是注冊一些本地方法供 Thread 類使用,如 start0(),stop0() 等等,可以說,所有操作本地線程的本地方法都是由它注冊的 . 這個方法放在一個 static 語句塊中,這就表明,當該類被加載到 JVM 中的時候,它就會被調(diào)用,進而注冊相應的本地方法。

private static native void registerNatives(); 
static{ 
 registerNatives(); 
}

本地方法 registerNatives 是定義在 Thread.c 文件中的。Thread.c 是個很小的文件,定義了各個操作系統(tǒng)平臺都要用到的關(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 的方法,實際上會調(diào)用到 JVM_StartThread 方法,那這個方法又是怎樣的邏輯呢。實際上,我們需要的是(或者說 Java 表現(xiàn)行為)該方法最終要調(diào)用 Java 線程的 run 方法,事實的確如此。 在 jvm.cpp 中,有如下代碼段:

JVM_ENTRY(void, JVM_StartThread(JNIEnv* env, jobject jthread)) 
 …
 native_thread = new JavaThread(&thread_entry, sz);

**這里JVM_ENTRY是一個宏,用來定義**JVM_StartThread 函數(shù),可以看到函數(shù)內(nèi)創(chuàng)建了真正的平臺相關(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 是如何聲明定義的,因為涉及到很繁瑣的代碼細節(jié),本文不做贅述。感興趣的讀者可以自行查看 JVM 的源代碼。

圖. Java 線程創(chuàng)建調(diào)用關(guān)系圖

這里寫圖片描述

start() 創(chuàng)建新進程
run() 沒有

PS:下面看下Java線程中run和start方法的區(qū)別

Thread類中run()和start()方法的區(qū)別如下:

run()方法:在本線程內(nèi)調(diào)用該Runnable對象的run()方法,可以重復多次調(diào)用;

start()方法:啟動一個線程,調(diào)用該Runnable對象的run()方法,不能多次啟動一個線程;

package com.ljq.test;
public class ThreadTest {
  /**
   * 觀察直接調(diào)用run()和用start()啟動一個線程的差別 
   * 
   * @param args
   * @throws Exception
   */
  public static void main(String[] args){
    Thread thread=new ThreadDemo();
    //第一種
    //表明: run()和其他方法的調(diào)用沒任何不同,main方法按順序執(zhí)行了它,并打印出最后一句
    //thread.run();
    //第二種
    //表明: start()方法重新創(chuàng)建了一個線程,在main方法執(zhí)行結(jié)束后,由于start()方法創(chuàng)建的線程沒有運行結(jié)束,
    //因此主線程未能退出,直到線程thread也執(zhí)行完畢.這里要注意,默認創(chuàng)建的線程是用戶線程(非守護線程)
    //thread.start();
    //第三種
    //1、為什么沒有打印出100句呢?因為我們將thread線程設置為了daemon(守護)線程,程序中只有守護線程存在的時候,是可以退出的,所以只打印了七句便退出了
    //2、當java虛擬機中有守護線程在運行的時候,java虛擬機會關(guān)閉。當所有常規(guī)線程運行完畢以后,
    //守護線程不管運行到哪里,虛擬機都會退出運行。所以你的守護線程最好不要寫一些會影響程序的業(yè)務邏輯。否則無法預料程序到底會出現(xiàn)什么問題
    //thread.setDaemon(true);
    //thread.start();
    //第四種
    //用戶線程可以被System.exit(0)強制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方法的操作技巧,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!

相關(guān)文章

  • Java中的EnumMap集合解析

    Java中的EnumMap集合解析

    這篇文章主要介紹了Java中的EnumMap集合解析,EnumMap是Map接口的一種實現(xiàn),專門用于枚舉類型的鍵,所有枚舉的鍵必須來自同一個枚舉,
    EnumMap不允許鍵為空,允許值為空,需要的朋友可以參考下
    2023-09-09
  • Java Collections集合繼承結(jié)構(gòu)圖_動力節(jié)點Java學院整理

    Java Collections集合繼承結(jié)構(gòu)圖_動力節(jié)點Java學院整理

    這篇文章主要介紹了Java Collections集合繼承結(jié)構(gòu)圖_動力節(jié)點Java學院整理,需要的朋友可以參考下
    2017-04-04
  • SpringBoot快速通關(guān)自動配置應用

    SpringBoot快速通關(guān)自動配置應用

    在進行項目編寫前,我們還需要知道一個東西,就是SpringBoot對我們的SpringMVC還做了哪些配置,包括如何擴展,如何定制,只有把這些都搞清楚了,我們在之后使用才會更加得心應手
    2022-07-07
  • Java Map.getOrDefault方法詳解

    Java Map.getOrDefault方法詳解

    Map.getOrDefault(Object key, V defaultValue)是Java中Map接口的一個方法,用于獲取指定鍵對應的值,如果鍵不存在,則返回一個默認值,這篇文章主要介紹了Java Map.getOrDefault方法詳解,需要的朋友可以參考下
    2024-01-01
  • JAVA日期處理類詳解

    JAVA日期處理類詳解

    這篇文章主要介紹了Java實現(xiàn)的日期處理類,結(jié)合完整實例形式分析了Java針對日期的獲取、運算、轉(zhuǎn)換等相關(guān)操作技巧,需要的朋友可以參考下
    2021-08-08
  • Spring注解驅(qū)動之關(guān)于@Bean注解指定初始化和銷毀的方法

    Spring注解驅(qū)動之關(guān)于@Bean注解指定初始化和銷毀的方法

    這篇文章主要介紹了Spring注解驅(qū)動之關(guān)于@Bean注解指定初始化和銷毀的方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-09-09
  • java讀取文件內(nèi)容的三種方法代碼片斷分享(java文件操作)

    java讀取文件內(nèi)容的三種方法代碼片斷分享(java文件操作)

    本文介紹java讀取文件內(nèi)容的三種方法,代碼可以直接放到程序中使用,大家參考使用吧
    2014-01-01
  • java中使用interrupt通知線程停止詳析

    java中使用interrupt通知線程停止詳析

    這篇文章主要介紹了java中使用interrupt通知線程停止詳析,文章介紹的是使用interrupt來通知線程停止運行,而不是強制停止,詳細內(nèi)容需要的小伙伴可以參考一下
    2022-09-09
  • elasticsearch如何根據(jù)條件刪除數(shù)據(jù)

    elasticsearch如何根據(jù)條件刪除數(shù)據(jù)

    Elasticsearch是一個基于Apache Lucene?的開源搜索引擎,無論在開源還是專有領(lǐng)域,Lucene 可以被認為是迄今為止最先進、性能最好的、功能最全的搜索引擎庫,這篇文章主要介紹了elasticsearch如何根據(jù)條件刪除數(shù)據(jù),需要的朋友可以參考下
    2023-03-03
  • Java中的FutureTask實現(xiàn)異步任務代碼實例

    Java中的FutureTask實現(xiàn)異步任務代碼實例

    這篇文章主要介紹了Java中的FutureTask實現(xiàn)異步任務代碼實例,普通的線程執(zhí)行是無法獲取到執(zhí)行結(jié)果的,FutureTask?間接實現(xiàn)了?Runnable?和?Future?接口,可以得到子線程耗時操作的執(zhí)行結(jié)果,AsyncTask?異步任務就是使用了該機制,需要的朋友可以參考下
    2024-01-01

最新評論