欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

每日六道java新手入門面試題,通往自由的道路--多線程

 更新時(shí)間:2021年07月01日 09:45:12   作者:太子爺哪吒  
這篇文章主要為大家分享了最有價(jià)值的6道多線程面試題,涵蓋內(nèi)容全面,包括數(shù)據(jù)結(jié)構(gòu)和算法相關(guān)的題目、經(jīng)典面試編程題等,對hashCode方法的設(shè)計(jì)、垃圾收集的堆和代進(jìn)行剖析,感興趣的小伙伴們可以參考一下

1. 你可以講下進(jìn)程與線程的區(qū)別?為什么要用多線程?

  • 進(jìn)程:進(jìn)程是程序的一次執(zhí)行過程,是系統(tǒng)運(yùn)行程序的基本單位。
  • 線程:單個(gè)進(jìn)程中執(zhí)行中每個(gè)任務(wù)就是一個(gè)線程。線程是進(jìn)程中執(zhí)行運(yùn)算的最小單位。
  • 區(qū)別
    • 一個(gè)程序至少有一個(gè)進(jìn)程,一個(gè)進(jìn)程至少有一個(gè)線程。
    • 一個(gè)線程只能屬于一個(gè)進(jìn)程,但是一個(gè)進(jìn)程可以擁有多個(gè)線程。多線程處理就是允許一個(gè)進(jìn)程中在同一時(shí)刻執(zhí)行多個(gè)任務(wù)即多個(gè)線程。
    • 每個(gè)獨(dú)立的進(jìn)程有程序運(yùn)行的入口、順序執(zhí)行序列和程序出口。但是線程不能獨(dú)立執(zhí)行,必須依存在應(yīng)用程序中,由應(yīng)用程序提供多個(gè)線程執(zhí)行控制,兩者均可并發(fā)執(zhí)行

為什么要用多線程:

  • 發(fā)揮多核CPU的優(yōu)勢,采用多線程的方式去同時(shí)完成幾件事情而不互相干擾。
  • 能夠有效的防止阻塞,多條線程同時(shí)運(yùn)行,哪怕一條線程的代碼執(zhí)行讀取數(shù)據(jù)阻塞,也不會(huì)影響其它任務(wù)的執(zhí)行。
  • 提高程序的效率。

2. 什么是上下文切換?

上下文切換一般發(fā)生在多線程情況下,因?yàn)橐粋€(gè) CPU 核心在任意時(shí)刻只能被一個(gè)線程使用,為了讓這些線程都能得到有效執(zhí)行,CPU 采取的策略是為每個(gè)線程分配時(shí)間片并輪轉(zhuǎn)的形式。而在多核cpu下,多線程是并行工作的,如果線程數(shù)多,單個(gè)核又會(huì)并發(fā)的調(diào)度線程,運(yùn)行時(shí)就會(huì)讓一個(gè)線程的時(shí)間片用完的時(shí)候就會(huì)重新處于就緒狀態(tài)讓給其他線程使用,這個(gè)過程就屬于上下文切換。

對于我們Java程序線程來說,一旦一個(gè)線程搶占到CPU資源的使用權(quán)后,另一個(gè)線程需要保存當(dāng)前的一個(gè)狀態(tài),以便下次搶占成功后可以回到當(dāng)前狀態(tài),JVM中有塊內(nèi)存地址叫程序計(jì)數(shù)器,用于記錄保存線程執(zhí)行到哪一行代碼,它是每個(gè)線程獨(dú)有的。執(zhí)行任務(wù)從保存到再次加載的過程就是上下文切換。

實(shí)際上,上下文切換也是對系統(tǒng)意味著來說會(huì)消耗大量的CPU時(shí)間,消耗大量資源。

以下幾種情況會(huì)發(fā)生上下文切換。

  • 線程的cpu時(shí)間片用完
  • 在發(fā)生垃圾回收的時(shí)候
  • 我們自己調(diào)用了 sleep、yield、wait、join、synchronized、lock 等方法

3. 說說你知道的幾種創(chuàng)建線程的方式

創(chuàng)建線程有以下方式:

繼承Thread類,重載它的run方法。

  • 在我們自己定義一個(gè)繼承于Thread類的子類,并重寫里面run方法,編寫相關(guān)邏輯代碼。
  • 在測試類中創(chuàng)建我剛自定義的線程子類對象
  • 調(diào)用子類實(shí)例的star方法來啟動(dòng)線程,通過start方法去調(diào)用到run方法里面的邏輯。

實(shí)現(xiàn) Runnalbe接口,重載 Runnalbe接口中的run方法實(shí)現(xiàn) 。

  • 我們定義一個(gè)實(shí)現(xiàn)Runnable接口實(shí)現(xiàn)類,并重寫里面的run方法
  • 在測試類中創(chuàng)建一個(gè)我們剛定義的接口實(shí)現(xiàn)類的實(shí)例,以實(shí)例對象作為target創(chuàng)建Thead對象,而得到的Thread對象就是我們線程子類對象。
  • 最后調(diào)用線程對象的start方法

實(shí)現(xiàn)Callable接口方式,重寫Callable接口中的call方法,并且這個(gè)call方法可以有返回值。

  • 我們定義一個(gè)實(shí)現(xiàn)創(chuàng)建實(shí)現(xiàn)Callable接口實(shí)現(xiàn)類,并重寫里面的call方法,注意它是call方法,并且有返回值。
  • 在測試類中創(chuàng)建一個(gè)我們剛定義的接口實(shí)現(xiàn)類的實(shí)例,以實(shí)例對象為參數(shù)創(chuàng)建FutureTask對象,并把創(chuàng)建出來FutureTask對象作為參數(shù)去創(chuàng)建Thread對象,而得到的Thread對象就是我們線程子類對象。
  • 最好調(diào)用線程對象的start方法。

需要注意三者的區(qū)別:

  • Thread是繼承,而Runnalbe、Callable是實(shí)現(xiàn)。對于繼承來說,只能單繼承,而接口可以多實(shí)現(xiàn)。如果繼承了 Thread類就無法再繼承其他類了。
  • 三者都是最后采用Thread.start()去啟動(dòng)線程,而不是調(diào)用run方法,或者call方法的。
  • Runnable接口 run 方法無返回值;Callable接口 call 方法有返回值。
  • Runnable 接口 run 方法只能拋出運(yùn)行時(shí)異常,且無法捕獲處理;Callable 接口 call 方法允許拋出異常,可以獲取異常信息
  • 使用實(shí)現(xiàn) Runnable接口的方式創(chuàng)建的線程可以處理同一資源,而實(shí)現(xiàn)資源的共享,還可以繼承其他類。

4. 昨天你講到創(chuàng)建線程后使用start方法去調(diào)用線程,為什么run方法不行呢?有什么區(qū)別?

我們先來看看代碼吧。

public class ThreadDemo {

    public static void main(String[] args) {
        MyThread myThread = new MyThread();
        MyThread myThead2 = new MyThread();
//        myThread.start();
//        myThead2.start();
        myThread.run();
        myThead2.run();
    }
}

class MyThread extends Thread {
    @Override
    public void run() {
        for (int i = 0; i < 6; i++) {
            System.out.println(Thread.currentThread().getName() + " :" + i);
            try {
                sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

這里我們創(chuàng)建了MyThread繼承了Thread類,這種方法是一種可以創(chuàng)建線程的方式。接著我們在main方法中創(chuàng)建了兩個(gè)線程,都調(diào)用了start方法和run方法。讓我們先看看結(jié)果吧!

// 注釋掉兩個(gè)run方法 開啟start方法得到的結(jié)果
Thread-0 :0
Thread-1 :0
Thread-1 :1
Thread-0 :1
Thread-1 :2
Thread-0 :2
Thread-1 :3
Thread-0 :3
Thread-1 :4
Thread-0 :4
Thread-1 :5
Thread-0 :5

// 注釋掉兩個(gè)start方法 開啟run方法得到的結(jié)果
main :0
main :1
main :2
main :3
main :4
main :5
main :0
main :1
main :2
main :3
main :4
main :5

接下來我們講一下:

1.start方法的作用:

啟動(dòng)線程,相當(dāng)于開啟一個(gè)線程調(diào)用我們重寫的run方法里面的邏輯,此時(shí)相當(dāng)于有兩個(gè)線程,一個(gè)main的主線程和開啟的子線程??梢钥吹轿覀兊拇a,相當(dāng)于有三個(gè)線程,一個(gè)主線程、一個(gè)Thread-0線程和一個(gè)Thread-1線程。并且線程之間是沒有順序的,他們是搶占cpu的資源來回切換的。

2.run方法的作用:

執(zhí)行線程的運(yùn)行時(shí)代碼,相當(dāng)于我們只是單純的調(diào)用一個(gè)普通方法。然后通過主線程的順序調(diào)用的方式,從myThread調(diào)用run方法結(jié)束后到myThread2去調(diào)用run方法結(jié)束,并且我們也可以看到我們控制臺(tái)中的線程名字就是main主線程。

3.run方法我們可以重復(fù)調(diào)用,而start方法在一個(gè)線程中只能調(diào)用一次。即myThread這個(gè)實(shí)例對象只能調(diào)用一次start方法,如果再調(diào)用一次start方法的話,就會(huì)拋出IllegalThreadStateException 的異常。

4.我們調(diào)用start方法算是真正意義上的多線程,因?yàn)樗穷~外開啟一個(gè)子線程去調(diào)用我們的run方法了。如果我們是調(diào)用run方法,就需要等待上一次的run方法執(zhí)行完畢才能調(diào)用下一次。所以我們要調(diào)用start方法充分揮多核CPU的優(yōu)勢,采用多線程的方式去同時(shí)完成幾件事情而不互相干擾。

5. 你知道你開啟一個(gè)線程后,它的狀態(tài)有那些嗎?

我們可以通過查看Thread的源碼中State枚舉發(fā)現(xiàn)有6個(gè)狀態(tài):

    public enum State {
        /**
         * Thread state for a thread which has not yet started.
         */
        NEW,

        /**
         * Thread state for a runnable thread.  A thread in the runnable
         * state is executing in the Java virtual machine but it may
         * be waiting for other resources from the operating system
         * such as processor.
         */
        RUNNABLE,

        /**
         * Thread state for a thread blocked waiting for a monitor lock.
         * A thread in the blocked state is waiting for a monitor lock
         * to enter a synchronized block/method or
         * reenter a synchronized block/method after calling
         * {@link Object#wait() Object.wait}.
         */
        BLOCKED,

        /**
         * Thread state for a waiting thread.
         * A thread is in the waiting state due to calling one of the
         * following methods:
         * <ul>
         *   <li>{@link Object#wait() Object.wait} with no timeout</li>
         *   <li>{@link #join() Thread.join} with no timeout</li>
         *   <li>{@link LockSupport#park() LockSupport.park}</li>
         * </ul>
         *
         * <p>A thread in the waiting state is waiting for another thread to
         * perform a particular action.
         *
         * For example, a thread that has called <tt>Object.wait()</tt>
         * on an object is waiting for another thread to call
         * <tt>Object.notify()</tt> or <tt>Object.notifyAll()</tt> on
         * that object. A thread that has called <tt>Thread.join()</tt>
         * is waiting for a specified thread to terminate.
         */
        WAITING,

        /**
         * Thread state for a waiting thread with a specified waiting time.
         * A thread is in the timed waiting state due to calling one of
         * the following methods with a specified positive waiting time:
         * <ul>
         *   <li>{@link #sleep Thread.sleep}</li>
         *   <li>{@link Object#wait(long) Object.wait} with timeout</li>
         *   <li>{@link #join(long) Thread.join} with timeout</li>
         *   <li>{@link LockSupport#parkNanos LockSupport.parkNanos}</li>
         *   <li>{@link LockSupport#parkUntil LockSupport.parkUntil}</li>
         * </ul>
         */
        TIMED_WAITING,

        /**
         * Thread state for a terminated thread.
         * The thread has completed execution.
         */
        TERMINATED;
    }

接下來我們具體來說說吧:

NEW(新建)

線程剛被創(chuàng)建,還只是一個(gè)實(shí)例對象,并未調(diào)用start方法啟動(dòng)。。MyThread myThread = new MyThread只有線程對象,沒有線程特征。

Runnable(可運(yùn)行)

在創(chuàng)建對象對象完成后,調(diào)用了myThread.start()方法線程,可以在Java虛擬機(jī)中運(yùn)行的狀態(tài),可能正在運(yùn)行自己代碼,也可能沒有,這取決于操作系統(tǒng)處理器。也可以叫做處于就緒狀態(tài),需要等待被線程調(diào)度選中,獲取cpu資源的使用權(quán)。

Teminated(被終止)

因?yàn)閞un方法正常退出而死亡,或者因?yàn)闆]有捕獲的異常終止了run方法而死亡。代表著此線程的生命周期結(jié)束了。

處于運(yùn)行狀態(tài)中的線程由于某種原因,暫時(shí)放棄對 CPU的使用權(quán),停止執(zhí)行,此時(shí)進(jìn)入阻塞狀態(tài),直到其進(jìn)入到就緒狀態(tài),才 有機(jī)會(huì)再次被 CPU 調(diào)用以進(jìn)入到運(yùn)行狀態(tài)。有以下三種相關(guān)阻塞狀態(tài):

Blocked(鎖阻塞)

當(dāng)一個(gè)線程試圖獲取一個(gè)對象鎖如(Synchronzied或Lock),而該對象鎖被其他的線程持有,則該線程進(jìn)入Blocked狀態(tài);只有當(dāng)該線程持有鎖時(shí),該線程將變成Runnable狀態(tài)。

Waiting(無限等待)

在調(diào)用了wait方法,JVM會(huì)把該線程放入等待隊(duì)列中,等待另一個(gè)線程執(zhí)行一個(gè)(喚醒),該線程此時(shí)狀態(tài)表示進(jìn)入Waiting狀態(tài)。進(jìn)入這個(gè)狀態(tài)后是不能自動(dòng)喚醒的,必須等待另一個(gè)線程調(diào)用notify或者notifyAll方法才能夠喚醒。

TimedWaiting(計(jì)時(shí)等待)

同waiting狀態(tài)一樣,調(diào)用sleep方法或者其他超時(shí)方法時(shí),他們將進(jìn)入Timed Waiting狀態(tài)。不過這一狀態(tài)只需保持到超時(shí)期滿或者接收到喚醒通知。

6. 既然講到超時(shí)方法,那你講下sleep和wait的區(qū)別和他們需要怎樣喚醒

sleep和wait方法他們都是可以暫停當(dāng)前線程的執(zhí)行,進(jìn)入一個(gè)阻塞狀態(tài)。

sleep:

我們可以指定睡眠時(shí)間,即讓程序暫停指定時(shí)間運(yùn)行,時(shí)間到了會(huì)繼續(xù)執(zhí)行代碼,如果時(shí)間未到我們想要換醒需要調(diào)用interrupt 方法來隨時(shí)喚醒即可。而調(diào)用interrupt 會(huì)使得sleep()方法拋出InterruptedException 異常,當(dāng)sleep()方法拋出異常我們就中斷了sleep的方法,從而讓程序繼續(xù)運(yùn)行下去。

wait:

調(diào)用該方法,可以導(dǎo)致線程進(jìn)入等待阻塞狀態(tài),會(huì)一直等待直到它被其他線程通過notify或者notifyAll方法喚醒。或者也可以使用wait(long timeout)表示時(shí)間到了自動(dòng)執(zhí)行,類似于sleep(long millis)。

notify():該方法會(huì)隨機(jī)選擇一個(gè)在該對象上調(diào)用wait方法的線程,解除其阻塞狀態(tài)。

notifyAll():該方法會(huì)喚醒所有的wait對象。

兩者的區(qū)別:

  • 兩者所屬的類不同:sleep是 Thread線程類的靜態(tài)方法;而wait是 Object類的方法。
  • 兩者是否是否鎖呢:sleep不釋放鎖;wait釋放鎖。
  • 兩者所使用的場景:sleep可以在任何需要的場景下調(diào)用;而wait必須使用在同步代碼塊或者同步方法中。
  • 兩者不同喚醒機(jī)制:sleep方法執(zhí)行睡眠時(shí)間完成后,線程會(huì)自動(dòng)蘇醒;而wait方法被調(diào)用后,線程不會(huì)自動(dòng)蘇醒,需要?jiǎng)e的線程調(diào)用同一個(gè)對象上的 notify或者 notifyAll方法,或者可以使用wait(long timeout)超時(shí)后線程會(huì)自動(dòng)蘇醒。

總結(jié):

這篇文章就到這里了,如果這篇文章對你也有所幫助,希望您能多多關(guān)注腳本之家的更多內(nèi)容!

相關(guān)文章

  • mybatis-flex與springBoot整合的實(shí)現(xiàn)示例

    mybatis-flex與springBoot整合的實(shí)現(xiàn)示例

    Mybatis-flex提供了簡單易用的API,開發(fā)者只需要簡單的配置即可使用,本文主要介紹了mybatis-flex與springBoot整合,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-01-01
  • SpringCloud集成Sleuth和Zipkin的思路講解

    SpringCloud集成Sleuth和Zipkin的思路講解

    Zipkin 是 Twitter 的一個(gè)開源項(xiàng)目,它基于 Google Dapper 實(shí)現(xiàn),它致力于收集服務(wù)的定時(shí)數(shù)據(jù),以及解決微服務(wù)架構(gòu)中的延遲問題,包括數(shù)據(jù)的收集、存儲(chǔ)、查找和展現(xiàn),這篇文章主要介紹了SpringCloud集成Sleuth和Zipkin,需要的朋友可以參考下
    2022-11-11
  • Java實(shí)現(xiàn)高效隨機(jī)數(shù)算法的示例代碼

    Java實(shí)現(xiàn)高效隨機(jī)數(shù)算法的示例代碼

    這篇文章主要介紹了Java實(shí)現(xiàn)高效隨機(jī)數(shù)算法的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2019-02-02
  • Java基礎(chǔ)夯實(shí)之線程問題全面解析

    Java基礎(chǔ)夯實(shí)之線程問題全面解析

    操作系統(tǒng)支持多個(gè)應(yīng)用程序并發(fā)執(zhí)行,每個(gè)應(yīng)用程序至少對應(yīng)一個(gè)進(jìn)程?。進(jìn)程是資源分配的最小單位,而線程是CPU調(diào)度的最小單位。本文將帶大家全面解析線程相關(guān)問題,感興趣的可以了解一下
    2022-11-11
  • 詳解為什么阿里巴巴禁止使用BigDecimal的equals方法做等值比較

    詳解為什么阿里巴巴禁止使用BigDecimal的equals方法做等值比較

    這篇文章主要介紹了詳解為什么阿里巴巴禁止使用BigDecimal的equals方法做等值比較,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-09-09
  • java工廠實(shí)例BeanFactoryPostProcessor和BeanPostProcessor區(qū)別分析

    java工廠實(shí)例BeanFactoryPostProcessor和BeanPostProcessor區(qū)別分析

    這篇文章主要為大家介紹了BeanFactoryPostProcessor和BeanPostProcessor區(qū)別示例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-07-07
  • springboot如何忽略接收請求中的參數(shù)

    springboot如何忽略接收請求中的參數(shù)

    這篇文章主要介紹了springboot如何忽略接收請求中的參數(shù)問題,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-07-07
  • Java索引越界異常Exception java.lang.IndexOutOfBoundsException的解決

    Java索引越界異常Exception java.lang.IndexOutOfBoundsException

    本文主要介紹了Java索引越界異常Exception java.lang.IndexOutOfBoundsException的解決,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • Java中的?stop?the?world是什么呢

    Java中的?stop?the?world是什么呢

    這篇文章主要介紹了Java中的stop?the?world是什么呢,從字面上講,就是停止這個(gè)世界,看到這個(gè)字眼,就覺得這是可怕的事情,那到底什么是stop-the-world,本文給大家詳細(xì)講解,感興趣的朋友跟隨小編一起看看吧
    2023-05-05
  • Java8 Predicate花樣用法詳解

    Java8 Predicate花樣用法詳解

    本文主要介紹了Java 8 Predicate花樣用法詳解,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03

最新評論