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

一文帶你學(xué)習(xí)Java中的線程

 更新時(shí)間:2023年05月15日 08:51:05   作者:真正的飛魚  
線程是系統(tǒng)調(diào)度的最小單元,一個(gè)進(jìn)程可以包含多個(gè)線程,線程是負(fù)責(zé)執(zhí)行二進(jìn)制指令的。本文將詳細(xì)給大家介紹一下Java中的線程,,需要的朋友可以參考下

介紹線程

線程是系統(tǒng)調(diào)度的最小單元,一個(gè)進(jìn)程可以包含多個(gè)線程,線程是負(fù)責(zé)執(zhí)行二進(jìn)制指令的。

每個(gè)線程有自己的程序計(jì)數(shù)器、棧(Stack)、寄存器(Register)、本地存儲(chǔ)(Thread Local)等,但是會(huì)和進(jìn)程內(nèi)其他線程共享文件描述符、虛擬地址空間等。

對(duì)于任何一個(gè)進(jìn)程來講,即便我們沒有主動(dòng)去創(chuàng)建線程,進(jìn)程也是默認(rèn)有一個(gè)主線程的。

守護(hù)線程(Daemon Thread)

有的時(shí)候應(yīng)用中需要一個(gè)長(zhǎng)期駐留的服務(wù)程序,但是不希望這個(gè)服務(wù)程序影響應(yīng)用退出,那么我們就可以將這個(gè)服務(wù)程序設(shè)置為守護(hù)線程,如果 Java 虛擬機(jī)發(fā)現(xiàn)只有守護(hù)線程存在時(shí),將結(jié)束進(jìn)程。

在 Java 中將線程設(shè)置為守護(hù)線程,具體的實(shí)現(xiàn)代碼如下所示:

public static void main(String[] args) {
    Thread daemonThread = new Thread();
    // 必須在線程啟動(dòng)之前設(shè)置
    daemonThread.setDaemon(true);
    daemonThread.start();
}

通用的線程生命周期

在操作系統(tǒng)層面,線程有生命周期。

對(duì)于有生命周期的事物,要學(xué)好它,只要能搞懂生命周期中各個(gè)節(jié)點(diǎn)的狀態(tài)轉(zhuǎn)換機(jī)制就可以了。

通用的線程生命周期基本上可以用下圖這個(gè) “五態(tài)模型” 來描述。這五態(tài)分別是:初始狀態(tài)、可運(yùn)行狀態(tài)、運(yùn)行狀態(tài)、休眠狀態(tài)和終止?fàn)顟B(tài)。

這“五態(tài)模型”的詳細(xì)情況如下所示。

初始狀態(tài)

初始狀態(tài),指的是線程已經(jīng)被創(chuàng)建,但是還不允許被 CPU 調(diào)度。

初始狀態(tài)屬于編程語言特有的,這里所謂的被創(chuàng)建,僅僅是在編程語言層面被創(chuàng)建,而在操作系統(tǒng)層面,真正的線程還沒有被創(chuàng)建。

在 Java 中,初始狀態(tài)相當(dāng)于是創(chuàng)建了 Thread 類的對(duì)象,但是還沒有調(diào)用 Thread#start() 方法。

可運(yùn)行狀態(tài)

可運(yùn)行狀態(tài),指的是線程可以被操作系統(tǒng)調(diào)度,但是線程還沒有開始執(zhí)行。

在可運(yùn)行狀態(tài)下,真正的操作系統(tǒng)線程已經(jīng)被創(chuàng)建。多個(gè)線程處于可運(yùn)行狀態(tài)時(shí),操作系統(tǒng)會(huì)根據(jù)調(diào)度算法選擇一個(gè)線程運(yùn)行。

在 Java 中,可運(yùn)行狀態(tài)相當(dāng)于是調(diào)用了 Thread#start() 方法,但是線程還沒有被分配 CPU 執(zhí)行。

運(yùn)行狀態(tài)

當(dāng)有空閑的 CPU 時(shí),操作系統(tǒng)會(huì)將空閑的 CPU 分配給一個(gè)處于可運(yùn)行狀態(tài)的線程,被分配到 CPU 的線程的狀態(tài)就從可運(yùn)行狀態(tài)轉(zhuǎn)換成了運(yùn)行狀態(tài)。

在 Java 中,運(yùn)行狀態(tài)相當(dāng)于是調(diào)用了 Thread#start() 方法,并且線程被分配 CPU 執(zhí)行。

休眠狀態(tài)

如果運(yùn)行狀態(tài)的線程調(diào)用了一個(gè)阻塞的 API(例如以阻塞的方式讀取文件)或者等待某個(gè)事件(例如條件變量),那么線程的狀態(tài)就會(huì)從運(yùn)行狀態(tài)轉(zhuǎn)換到休眠狀態(tài),同時(shí)釋放 CPU 的使用權(quán),休眠狀態(tài)的線程永遠(yuǎn)沒有機(jī)會(huì)獲得 CPU 的使用權(quán)。

當(dāng)?shù)却馁Y源或條件滿足后,線程就會(huì)從休眠狀態(tài)轉(zhuǎn)換到可運(yùn)行狀態(tài),并等待 CPU 調(diào)度。

終止?fàn)顟B(tài)

線程執(zhí)行完畢或者出現(xiàn)異常,線程就會(huì)進(jìn)入終止?fàn)顟B(tài),即線程的生命周期終止。

這五種狀態(tài)在不同編程語言里會(huì)有簡(jiǎn)化合并。例如:

  • C 語言的 POSIX Threads 規(guī)范,就把初始狀態(tài)和可運(yùn)行狀態(tài)合并了;
  • Java 程序設(shè)計(jì)語言把可運(yùn)行狀態(tài)和運(yùn)行狀態(tài)合并了,這兩個(gè)狀態(tài)在操作系統(tǒng)調(diào)度層面有用,而 Java 虛擬機(jī)層面不關(guān)心這兩個(gè)狀態(tài),因?yàn)?Java 虛擬機(jī)把線程調(diào)度交給操作系統(tǒng)處理了。

除了簡(jiǎn)化合并,這五種狀態(tài)也有可能被細(xì)化,比如,Java 語言里就細(xì)化了休眠狀態(tài)(這個(gè)下面我們會(huì)詳細(xì)講解)。

Java 的線程生命周期

不同的程序設(shè)計(jì)語言對(duì)于操作系統(tǒng)線程進(jìn)行了不同的封裝,下面我們學(xué)習(xí)一下 Java 的線程生命周期。

Java 程序設(shè)計(jì)語言中,線程共有六種狀態(tài),分別是:

  • NEW(初始狀態(tài))
  • RUNNABLE(可運(yùn)行 / 運(yùn)行狀態(tài))
  • BLOCKED(阻塞狀態(tài))
  • WAITING(無時(shí)限等待)
  • TIMED_WAITING(有時(shí)限等待)
  • TERMINATED(終止?fàn)顟B(tài))

NEW(初始狀態(tài))、TERMINATED(終止?fàn)顟B(tài))和通用的線程生命周期中的語義相同。

在操作系統(tǒng)層面,Java 線程中的 BLOCKED、WAITING、TIMED_WAITING 是一種狀態(tài),即通用的線程生命周期中的休眠狀態(tài)。也就是說只要 Java 線程處于這三種狀態(tài)之一,那么這個(gè)線程就永遠(yuǎn)沒有機(jī)會(huì)獲得 CPU 的使用權(quán)。

所以 Java 中的線程生命周期可以簡(jiǎn)化為下圖:

其中,可以將 BLOCKED、WAITING、TIMED_WAITING 理解為導(dǎo)致線程處于休眠狀態(tài)的三種原因。

  • 那具體是哪些情形會(huì)導(dǎo)致線程從 RUNNABLE 狀態(tài)轉(zhuǎn)換到這三種狀態(tài)呢?
  • 而這三種狀態(tài)又是何時(shí)轉(zhuǎn)換回 RUNNABLE 的呢?
  • 以及 NEW、TERMINATED 和 RUNNABLE 狀態(tài)是如何轉(zhuǎn)換的?

下面我們?cè)敿?xì)講解。

Java 的線程狀態(tài)切換

從 NEW 到 RUNNABLE 狀態(tài)

剛創(chuàng)建 Thread 類的對(duì)象時(shí),線程處于 NEW 狀態(tài)。

NEW 狀態(tài)的線程,不會(huì)被操作系統(tǒng)調(diào)度,因此不會(huì)執(zhí)行。Java 線程要執(zhí)行,就必須轉(zhuǎn)換到 RUNNABLE 狀態(tài)。

從 NEW 狀態(tài)轉(zhuǎn)換到 RUNNABLE 狀態(tài)只要調(diào)用線程對(duì)象的 start() 方法就可以了,具體的實(shí)現(xiàn)代碼如下所示:

public static void main(String[] args) {
    Thread thread = new Thread(new Runnable() {
        @Override
        public void run() {
            System.out.println("hello");
        }
    });
    thread.start();
}

從 RUNNABLE 到 TERMINATED 狀態(tài)

線程執(zhí)行完 Thrad#run() 方法后,會(huì)自動(dòng)從 RUNNABLE 狀態(tài)轉(zhuǎn)換到 TERMINATED 狀態(tài)。

如果執(zhí)行 run() 方法的時(shí)候異常了拋出,也會(huì)導(dǎo)致線程終止,進(jìn)入 TERMINATED 狀態(tài) 。

1. RUNNABLE 與 BLOCKED 的狀態(tài)轉(zhuǎn)換

只有一種場(chǎng)景會(huì)觸發(fā) RUNNABLE 與 BLOCKED 的狀態(tài)轉(zhuǎn)換,就是線程等待 synchronized 的隱式鎖。

  • 當(dāng)使用 synchronized 申請(qǐng)加鎖失敗時(shí),該線程的狀態(tài)就會(huì)從 RUNNABLE 轉(zhuǎn)換到 BLOCKED 狀態(tài)。
  • 當(dāng)?shù)却木€程獲得鎖時(shí),該線程的狀態(tài)就會(huì)從 BLOCKED 狀態(tài)轉(zhuǎn)換到 RUNNABLE 狀態(tài)。

如果你熟悉操作系統(tǒng)線程的生命周期的話,可能會(huì)有個(gè)疑問:線程調(diào)用阻塞式 API 時(shí),是否會(huì)轉(zhuǎn)換到 BLOCKED 狀態(tài)呢?在操作系統(tǒng)層面,線程是會(huì)轉(zhuǎn)換到休眠狀態(tài)的,但是在 Java 虛擬機(jī)層面,Java 線程的狀態(tài)不會(huì)發(fā)生變化,也就是說 Java 線程的狀態(tài)會(huì)依然保持 RUNNABLE 狀態(tài)。

Java 虛擬機(jī)層面并不關(guān)心操作系統(tǒng)調(diào)度相關(guān)的狀態(tài),因?yàn)樵?Java 虛擬機(jī)看來,等待 CPU 的使用權(quán)(操作系統(tǒng)層面此時(shí)處于可執(zhí)行狀態(tài))與等待 I/O(操作系統(tǒng)層面此時(shí)處于休眠狀態(tài))沒有區(qū)別,都是在等待某個(gè)資源,所以都?xì)w入了 RUNNABLE 狀態(tài)。

而我們說的 Java 線程在調(diào)用阻塞式 API 時(shí),線程會(huì)阻塞,指的是操作系統(tǒng)線程的狀態(tài),并不是 Java 線程的狀態(tài)。

2. RUNNABLE 與 WAITING 的狀態(tài)轉(zhuǎn)換

總體來說,有三種場(chǎng)景會(huì)觸發(fā) RUNNABLE 與 WAITING 的狀態(tài)轉(zhuǎn)換。

第一種場(chǎng)景,獲得 synchronized 隱式鎖的線程,調(diào)用無參數(shù)的 Object#wait() 方法。

這里應(yīng)該調(diào)用的是鎖對(duì)象的 wait() 方法,具體的實(shí)現(xiàn)代碼如下所示:

public void method() throws InterruptedException {
    synchronized (this) {
        this.wait();
    }
}
  • 當(dāng)調(diào)用 wait() 方法時(shí),調(diào)用方法的線程的狀態(tài)從 RUNNABLE 狀態(tài)轉(zhuǎn)換到 WAITING 狀態(tài)
  • 當(dāng)調(diào)用 notify() 方法時(shí),被喚醒的線程的狀態(tài)從 WAITING 狀態(tài)轉(zhuǎn)換到 RUNNABLE 狀態(tài)

第二種場(chǎng)景,調(diào)用無參數(shù)的 Thread#join() 方法。

join() 是一種線程同步方法,例如有一個(gè)線程對(duì)象 thread A:

  • 當(dāng)調(diào)用 A.join() 方法時(shí),執(zhí)行這條語句的線程會(huì)等待 thread A 執(zhí)行完,而等待中的這個(gè)線程,其狀態(tài)會(huì)從 RUNNABLE 轉(zhuǎn)換到 WAITING。
  • 當(dāng)線程 thread A 執(zhí)行完,原來等待它的線程又會(huì)從 WAITING 狀態(tài)轉(zhuǎn)換到 RUNNABLE。

Thread#join() 方法的實(shí)現(xiàn)基于 Object#wait()。

第三種場(chǎng)景,調(diào)用 LockSupport#park() 方法。

LockSupport 類,也許你有點(diǎn)陌生,其實(shí) Java 并發(fā)包中鎖的實(shí)現(xiàn)都用到了 LockSupport#park() / unpark()。

  • 當(dāng)調(diào)用 LockSupport.park() 方法時(shí),調(diào)用方法的線程的狀態(tài)從 RUNNABLE 轉(zhuǎn)換到 WAITING。
  • 當(dāng)調(diào)用 LockSupport.unpark(Thread thread) 方法時(shí),被喚醒的線程的狀態(tài)從 WAITING 狀態(tài)轉(zhuǎn)換到 RUNNABLE 狀態(tài)

總結(jié)來說:Object#wait() 和 LockSupport#park() 方法使線程的狀態(tài)轉(zhuǎn)換到 WAITING。

3. RUNNABLE 與 TIMED_WAITING 的狀態(tài)轉(zhuǎn)換

總體來說,有五種場(chǎng)景會(huì)觸發(fā) RUNNABLE 與 TIMED_WAITING 的狀態(tài)轉(zhuǎn)換:

  • 獲得 synchronized 隱式鎖的線程,調(diào)用帶超時(shí)參數(shù)的 Object#wait(long timeout) 方法;
  • 調(diào)用帶超時(shí)參數(shù)的 Thread#join(long millis) 方法;(底層調(diào)用 Object#wait(long timeout) )
  • 調(diào)用帶超時(shí)參數(shù)的 LockSupport.parkNanos(Object blocker, long deadline) 方法;
  • 調(diào)用帶超時(shí)參數(shù)的 LockSupport.parkUntil(long deadline) 方法。
  • 調(diào)用帶超時(shí)參數(shù)的 Thread.sleep(long millis) 方法;

這里你會(huì)發(fā)現(xiàn):

  • TIMED_WAITING 和 WAITING 狀態(tài)的區(qū)別,僅僅是觸發(fā)條件多了超時(shí)參數(shù)。
  • 與 RUNNABLE 與 WAITING 的狀態(tài)轉(zhuǎn)換 相比,多了一個(gè) Thread.sleep() 場(chǎng)景。

Java 線程 API 的使用

線程的創(chuàng)建

創(chuàng)建線程的幾種方式:

  • 繼承 Thread 類,重寫 run() 方法。
  • 實(shí)現(xiàn) Runnable 接口,實(shí)現(xiàn)其中的 run() 方法。將該實(shí)現(xiàn)類的對(duì)象作為參數(shù)傳遞到 Thread 類的構(gòu)造器中,創(chuàng)建 Thread 類的對(duì)象。
  • 實(shí)現(xiàn) Callable 接口,實(shí)現(xiàn)其中的 call() 方法。將該實(shí)現(xiàn)類的對(duì)象作為參數(shù)傳遞到 FutureTask 類的構(gòu)造器中,創(chuàng)建FutureTask 類的對(duì)象。將 FutureTask 類的對(duì)象作為參數(shù)傳遞到 Thread 類的構(gòu)造器中,創(chuàng)建 Thread 類的對(duì)象。Callable 它解決了 Runnable 無法返回結(jié)果的困擾。

「實(shí)現(xiàn) Runnable 接口」VS「繼承 Thread 類」

  • 通過實(shí)現(xiàn)(implements)的方式?jīng)]有類的單繼承性的局限性
  • 實(shí)現(xiàn)的方式更適合處理多個(gè)線程有共享數(shù)據(jù)的情況

「實(shí)現(xiàn) Callable 接口」VS「實(shí)現(xiàn) Runnable 接口」

  • call() 可以有返回值
  • call() 可以拋出異常被外面的操作捕獲,獲取異常的信息
  • 「實(shí)現(xiàn) Callable 接口」支持泛型
// 自定義線程對(duì)象
class MyThread extends Thread {
    public void run() {
        // 線程需要執(zhí)行的代碼
        ......
    }
}

// 創(chuàng)建線程對(duì)象
MyThread myThread = new MyThread();
// 實(shí)現(xiàn)Runnable接口
class Runner implements Runnable {
    @Override
    public void run() {
        // 線程需要執(zhí)行的代碼
        ......
    }
}

// 創(chuàng)建線程對(duì)象
Thread thread = new Thread(new Runner());
public static void main(String[] args) throws ExecutionException, InterruptedException {
    MyTask task = new MyTask();
    // FutureTask 用于接收運(yùn)算結(jié)果
    FutureTask futureTask = new FutureTask<>(task);
    Thread thread = new Thread(futureTask);

    thread.start();
	// FutureTask 可用于線程間同步 (當(dāng)前線程等待其他線程執(zhí)行完成之后,當(dāng)前線程才繼續(xù)執(zhí)行)
    // get() 返回值即為 FutureTask 構(gòu)造器參數(shù) Callable 實(shí)現(xiàn)類實(shí)現(xiàn)的 call() 的返回值
    System.out.println(futureTask.get());
}

public class MyTask implements Callable {
    @Override
    public String call() {
        // 若不需要返回值,可 return null;
        return "ok";
    }
}

線程的執(zhí)行

創(chuàng)建好 Thread 類的對(duì)象后,通過調(diào)用 Thread#start() 方法創(chuàng)建線程執(zhí)行任務(wù)。

線程執(zhí)行要調(diào)用 start() 而不是直接調(diào)用 run(),直接調(diào)用 run() 方法只會(huì)在當(dāng)前線程上同步執(zhí)行 run() 方法的內(nèi)容,而不會(huì)啟動(dòng)新線程。調(diào)用 start() 方法的作用:

  • 啟動(dòng)一個(gè)新的線程
  • 新的線程調(diào)用 run() 方法

線程的停止

有時(shí)候我們需要強(qiáng)制中斷 run() 方法的執(zhí)行,例如 run() 方法訪問一個(gè)很慢的網(wǎng)絡(luò),我們等不下去了,想終止怎么辦呢?Java 的 Thread 類里面倒是有個(gè) stop() 方法,不過已經(jīng)標(biāo)記為 @Deprecated,所以不建議使用了。正確的方式是調(diào)用 interrupt() 方法。Thread#interrupt() 配合合適的代碼,即可優(yōu)雅的實(shí)現(xiàn)線程的終止。

stop() 和 interrupt() 方法的區(qū)別。

  • stop() 方法會(huì)真的殺死線程,不給線程喘息的機(jī)會(huì),如果線程持有 ReentrantLock 鎖,被 stop() 的線程并不會(huì)自動(dòng)調(diào)用 ReentrantLock 的 unlock() 去釋放鎖,那其他線程就再也沒機(jī)會(huì)獲得 ReentrantLock 鎖,這實(shí)在是太危險(xiǎn)了。所以該方法就不建議使用了,類似的方法還有 suspend() 和 resume() 方法,這兩個(gè)方法同樣也都不建議使用了。
  • interrupt() 方法僅僅是通知線程,線程有機(jī)會(huì)執(zhí)行一些后續(xù)操作,線程也可以無視這個(gè)通知。被 interrupt 的線程,是怎么收到通知的呢?一種是異常,另一種是主動(dòng)檢測(cè)。

異常

當(dāng)線程 A 處于 WAITING、TIMED_WAITING 狀態(tài)時(shí),如果其他的線程調(diào)用線程 A 的 interrupt() 方法,會(huì)使線程 A 返回到 RUNNABLE 狀態(tài),同時(shí)線程 A 的代碼會(huì)觸發(fā) InterruptedException 異常。

上面我們提到轉(zhuǎn)換到 WAITING、TIMED_WAITING 狀態(tài)的觸發(fā)條件,都是調(diào)用了類似 wait()、join()、sleep() 這樣的方法,我們看這些方法的簽名,發(fā)現(xiàn)都會(huì) throws InterruptedException 這個(gè)異常。這個(gè)異常的觸發(fā)條件就是:其他的線程調(diào)用了該線程的 interrupt() 方法。

當(dāng)線程 A 處于 RUNNABLE 狀態(tài)時(shí):

  • 當(dāng)線程 A 處于 RUNNABLE 狀態(tài),并且阻塞在 java.nio.channels.InterruptibleChannel 上時(shí),如果其他的線程調(diào)用線程 A 的 interrupt() 方法,線程 A 會(huì)觸發(fā) java.nio.channels.ClosedByInterruptException 這個(gè)異常;
  • 當(dāng)線程 A 處于 RUNNABLE 狀態(tài),并且阻塞在 java.nio.channels.Selector 上時(shí),如果其他的線程調(diào)用線程 A 的 interrupt() 方法,線程 A 的 java.nio.channels.Selector 會(huì)立即返回。

上面這兩種情況屬于被中斷的線程通過異常的方式獲得了通知。

主動(dòng)檢測(cè)

還有一種是主動(dòng)檢測(cè),如果線程處于 RUNNABLE 狀態(tài),并且沒有阻塞在某個(gè) I/O 操作上,例如中斷計(jì)算圓周率的線程 A,這時(shí)就得依賴線程 A 主動(dòng)檢測(cè)中斷狀態(tài)了。如果其他的線程調(diào)用線程 A 的 interrupt() 方法,那么線程 A 可以通過 isInterrupted() 方法,檢測(cè)是不是自己被中斷了。

以上就是一文帶你學(xué)習(xí)Java中的線程的詳細(xì)內(nèi)容,更多關(guān)于Java 線程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 通過反射注解批量插入數(shù)據(jù)到DB的實(shí)現(xiàn)方法

    通過反射注解批量插入數(shù)據(jù)到DB的實(shí)現(xiàn)方法

    今天小編就為大家分享一篇關(guān)于通過反射注解批量插入數(shù)據(jù)到DB的實(shí)現(xiàn)方法,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧
    2019-03-03
  • SpringBoot使用 druid 連接池來優(yōu)化分頁語句

    SpringBoot使用 druid 連接池來優(yōu)化分頁語句

    這篇文章主要介紹了SpringBoot使用 druid 連接池來優(yōu)化分頁語句,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-11-11
  • Java中Calendar類的一些常用方法小結(jié)

    Java中Calendar類的一些常用方法小結(jié)

    項(xiàng)目當(dāng)中,我們經(jīng)常會(huì)涉及到對(duì)時(shí)間的處理,Date類最主要的作用就是獲得當(dāng)前時(shí)間,同時(shí)這個(gè)類里面也具有設(shè)置時(shí)間以及一些其他的功能,但更推薦使用 Calendar 類進(jìn)行時(shí)間和日期的處理,這篇文章主要給大家介紹了關(guān)于Java中Calendar類的一些常用方法,需要的朋友可以參考下
    2021-11-11
  • java中文分詞之正向最大匹配法實(shí)例代碼

    java中文分詞之正向最大匹配法實(shí)例代碼

    中文分詞應(yīng)用很廣泛,網(wǎng)上也有很多開源項(xiàng)目,下面這篇文章主要給大家介紹了關(guān)于java中文分詞之正向最大匹配法的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-11-11
  • java+mysql實(shí)現(xiàn)商品搶購功能

    java+mysql實(shí)現(xiàn)商品搶購功能

    這篇文章主要為大家詳細(xì)介紹了java+mysql實(shí)現(xiàn)商品搶購功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-02-02
  • Java實(shí)現(xiàn)大文件的分片上傳與下載(springboot+vue3)

    Java實(shí)現(xiàn)大文件的分片上傳與下載(springboot+vue3)

    這篇文章主要為大家詳細(xì)介紹了java基于springboot+vue3如何大文件的分片上傳與下載,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2023-06-06
  • MyBatis解決Update動(dòng)態(tài)SQL逗號(hào)的問題

    MyBatis解決Update動(dòng)態(tài)SQL逗號(hào)的問題

    這篇文章主要介紹了MyBatis解決Update動(dòng)態(tài)SQL逗號(hào)的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-01-01
  • Spring?Security?過濾器注冊(cè)脈絡(luò)梳理

    Spring?Security?過濾器注冊(cè)脈絡(luò)梳理

    這篇文章主要介紹了Spring?Security過濾器注冊(cè)脈絡(luò)梳理,Spring?Security在Servlet的過濾鏈中注冊(cè)了一個(gè)過濾器FilterChainProxy,它會(huì)把請(qǐng)求代理到Spring?Security自己維護(hù)的多個(gè)過濾鏈,每個(gè)過濾鏈會(huì)匹配一些URL,如果匹配則執(zhí)行對(duì)應(yīng)的過濾器
    2022-08-08
  • java實(shí)現(xiàn)文件復(fù)制、剪切文件和刪除示例

    java實(shí)現(xiàn)文件復(fù)制、剪切文件和刪除示例

    這篇文章主要介紹了java實(shí)現(xiàn)文件復(fù)制、剪切文件和刪除示例,需要的朋友可以參考下
    2014-04-04
  • 基于Maven導(dǎo)入pom依賴很慢的解決方案

    基于Maven導(dǎo)入pom依賴很慢的解決方案

    這篇文章主要介紹了Maven導(dǎo)入pom依賴很慢的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10

最新評(píng)論