主頁 > 後端開發 > Java多執行緒通關——基礎知識挑戰

Java多執行緒通關——基礎知識挑戰

2020-10-17 21:31:18 後端開發

等掌握了基礎知識之后,才有資格說基礎知識沒用這樣的話,否則就老老實實的開始吧,

 

 

物件的監視器


每一個Java物件都有一個監視器,并且規定,每個物件的監視器每次只能被一個執行緒擁有,只有擁有它的執行緒把它釋放之后,這個監視器才會被其它執行緒擁有,

其實就是說,物件的監視器對于多執行緒來說是互斥的,即一個執行緒從拿到它之后到釋放它之前這段時間內,其它執行緒是絕對不可能再拿到它的,這是由JVM保證的,

這樣一來,
物件的監視器就可以用來保護那種每次只允許一個執行緒執行的方法或代碼片段,就是我們常說的同步方法或同步代碼塊,

Java包括兩種范疇的物件(當然,這樣講可能不準確,主要用于幫助理解),
一種就是普通的物件,比如new Object(),一種就是描述型別資訊的物件,即Class<?>型別的物件,

這兩類都是Java物件,這毋庸置疑,所以它們都有監視器,但這兩類物件又有明顯的不同,所以它們的監視器對外表現的行為也是不同的,

請看下面運算式:

 

Object o1 = new Object();Object o2 = new Object();o1 == o2; //false

 

o1和o2是分別new出來的兩個物件,它們肯定不相同,又因為監視器是和物件關聯的,所以o1的監視器和o2的監視器也是不同的,且它們沒有任何關系,

所以
必須是同一個物件的監視器才行,不同物件的監視器達不到預期的效果,這一點要切記,

再看下面的運算式:

 

o1.getClass() == o2.getClass(); //trueo1.getClass() == Object.class; //true

 

但是o1的型別資訊物件(o1.getClass())和o2的型別資訊物件(o2.getClass())是同一個,且和Object類的型別資訊物件(Object.class)也是同一個,這不廢話嘛,o1和o2都是從Object類new出來的,哈哈,

型別資訊物件本身的型別是Class<?>,
在類加載器(ClassLoader)加載一個類后,就會生成一個和該類相關的Class<?>型別的物件,該物件會被快取起來,所以型別資訊物件是全域(同一個JVM同一個類加載器)唯一的,

這也就說明了,為什么同一個類new出來的多個物件是不同的,但是它們的型別資訊物件卻是同一個,且可以使用“類.class”直接獲取到它,

Java語言規定,
使用synchronized關鍵字可以獲取物件的監視器,下面分別來看這兩類物件的監視器用法,

普遍物件的監視器:

 

class SyncA {  //方法A  public synchronized void methodA() {    //同步方法,當前物件的監視器  }  //方法B  public void methodB() {    synchronized(this) {    //同步代碼塊,當前物件的監視器    }  }}
class SyncB { //物件 private SyncA syncA; public SyncB(SyncA syncA) { this.syncA = syncA; } //方法C public void methodC() { synchronized(syncA) { //同步代碼塊,syncA物件的監視器 } }}
//new一個物件SyncA syncA = new SyncA();//把該物件傳進去SyncB syncB = new SyncB(syncA);//A、B、C這三個方法都要擁有syncA這個物件的監視器才能執行new Thread(syncA::methodA).start();new Thread(syncA::methodB).start();new Thread(syncB::methodC).start();

 

這三個執行緒都去獲取同一個物件(即syncA)的監視器,因為一個物件的監視器一次只能被一個執行緒擁有,所以這三個執行緒是逐次獲取到的,因此這三個方法也是逐次執行的,

這個示例告訴我們,
利用物件的監視器可以做到的,并不只是同一個方法不能同時被多個執行緒執行,多個不同的方法也可以不能同時被多個執行緒執行,只要它們用到的是同一個物件的監視器,

型別資訊物件的監視器:

 

class SyncC {  //靜態方法A  public static synchronized void methodA() {    //同步方法,型別資訊物件的監視器  }  //靜態方法B  public static void methodB() {    synchronized(SyncC.class) {    //同步代碼塊,型別資訊物件的監視器    }  }}
class SyncD { //型別資訊物件 private Class<SyncC> syncClass; public SyncD(Class<SyncC> syncClass) { this.syncClass = syncClass; } //方法C public void methodC() { synchronized(syncClass) { //同步代碼塊,SyncC類的型別資訊物件的監視器 } } //方法D public void methodD() { synchronized(syncClass) { //同步代碼塊,SyncC類的型別資訊物件的監視器 } }}
//A、B、C、D這四個方法都要擁有SyncC類的型別資訊物件的監視器才能執行new Thread(SyncC::methodA).start();new Thread(SyncC::methodB).start();new Thread(new SyncD(SyncC.class)::methodC).start();new Thread(new SyncD((Class<SyncC>)new SyncC().getClass())::methodD).start();

 

因為一個類的型別資訊物件只有一個,所以這四個執行緒其實是在競爭同一個物件的監視器,因此這四個方法也是逐次執行的,

通過這個示例,再次強調一下,
不管是方法還是代碼塊,不管是靜態的還是實體的,也不管是屬于同一個類的還是多個類的,只要它們共用同一個物件的監視器,那么這些方法或代碼塊在多執行緒下是無法并發運行的,只能逐個運行,因為同一個物件的監視器每次只能被一個執行緒所擁有,其它執行緒此時只能被阻塞著,

注:
在實際使用中,一定要確保是同一個物件,尤其是使用字串型別或數字型別的物件時,一定要注意


幾個重要的方法


首先是Object類的
wait/notify/notifyAll方法,因為Java中的所有類最終都繼承自Object類,所以,可以使用任何Java物件來呼叫這三個方法,

不過Java規定,
要在某個物件上呼叫這三個方法,必須先獲取那個物件的監視器才行,再次提醒,監視器是和物件關聯的,不同的物件監視器也是不同的,

請看下面的用法:

 

//new一個物件Object obj = new Object();//獲取物件的監視器synchronized(obj) {  //在物件上呼叫wait方法  obj.wait();}

 

很多人首次接觸這一部分的時候一般都會比較懵,主要是因為搞不清人物關系,

這里的
wait方法雖然是在物件(即obj)上呼叫的,但卻不是讓這個物件等待的,而是讓執行這行代碼(即obj.wati())的執行緒(即Thread)在這個物件(即obj)上等待的,

這里的
執行緒是等待的“主體”,物件是等待的“位置”,比如學校開運動會時,會在操場上為每班劃定一個位置,并插上一個牌子,寫上班級名稱

這個
牌子就相當于物件obj,它表示一個位置資訊,當學生看到本班牌子之后,就會自動去牌子后面排隊等待,

學生就相當于執行緒,當學生看到牌子就相當于當執行緒執行到obj.wait(),學生去牌子后面排隊等待就相當于執行緒在物件obj上等待,

執行緒執行完obj.wait()后,就會釋放掉物件obj的監視器,轉而進入物件obj的等待集合中進行等待,執行緒由運行狀態變為等待(WAITING)狀態,此后這個執行緒將不再被執行緒調度器調度

(說明一點,當
多個執行緒去競爭同一個物件的監視器而沒有競爭上時,執行緒會變為阻塞(BLOCKED)狀態,而非等待狀態,)

執行緒選擇等待的原因大多都是因為需要的資源暫時得不到,那什么時候資源能就位讓執行緒再次執行呢?其實是不太好確定的,那干脆就到資源OK時通知它一聲吧,

請看下面的方法:

 

//獲取物件(還是上面那個)的監視器synchronized(obj) {  //在物件上呼叫notify方法  obj.notify();}

 

有了上面的基礎,現在就好理解多了,代碼的意思就是通知在物件obj上等待的執行緒,把其中一個喚醒,即把這個執行緒從物件obj的等待集合中移除,此后這個執行緒就又可以被執行緒調度器調度了,可能有一部分人覺得現在被喚醒的那個執行緒就可以執行了,其實不然

當前執行緒執行完notify方法后,必須要釋放掉物件obj的監視器,這樣被喚醒的那個執行緒才能重新獲取物件obj的監視器,這樣才可以繼續執行,

就是
當一個執行緒想要通過wait進入等待時,需要獲取物件的監視器,當別的執行緒通過notify喚醒這個執行緒時,這個執行緒想要繼續執行,還需要獲取物件的監視器

notifyAll方法的用法和notify是一樣的,只是含義不同,表示通知物件obj上所有等待的執行緒,把它們全部都喚醒,雖然是全部喚醒,但也只能有一個執行緒可以運行,因為每次只有一個執行緒能獲取到物件obj的監視器,

還有一種wait方法是帶有超時時間的,它表示執行緒進入等待的時間達到超時時間后還沒有被喚醒時,它會自動醒來(也可以認為是被系統喚醒的),

這種情況下沒有超時例外拋出,
雖然執行緒是自動醒來,但想要繼續執行的話,同樣需要先獲取物件obj的監視器才行

注:
執行緒通過wait進入等待時,只會釋放和這個wait相關的那個物件的監視器,如果此時執行緒還擁有其它物件的監視器,并不會去釋放它們,而是在等待期間一直擁有,這塊一定要注意,避免死鎖

使用須知:

處在等待狀態的執行緒,可能會被意外喚醒,即此時條件并不滿足,但是卻被喚醒了,當然,這種情況在實踐中很少發生,但是我們還是要做一些措施來應對,那就是再次檢測條件是否滿足,不滿足的話再次進入等待

可見這是一個具有重復性的邏輯,因此
把它放到一個回圈里是最合適的,如下這樣:

 

//獲取物件的監視器synchronized(obj) {  //判斷條件是否滿足  while(condition is not satisfied) {    //在物件上呼叫wait方法    obj.wait();  }}

 

這樣一來,即使被意外喚醒,還會再次進入等待,直到條件滿足后,才會退出while回圈,執行后面的邏輯

多執行緒的話題怎么能少了主角呢,下面有請主角上場,哈哈,就是Thread類啦,關于執行緒,我在上一篇文章中已經談過,這里再贅述一遍,希望加深一下印象,

執行緒是可以獨立運行的“個體”,這就導致我們對它的“控制能力”變弱了,當我們想讓一個執行緒暫停或停止時,如果強制去執行,會產生兩方面的問題,
一是使正在執行的業務中斷,導致業務出現不一致性,二是使正在使用的資源得不到釋放,導致記憶體泄漏或死鎖,可見,強制這種方式不可取,(看看Thread類的那些廢棄方法便知)

所以,只能采取柔和的方式,就是你對一個執行緒說,“大哥,停下來歇會吧”,或者是,“大哥,停止吧,不用再執行了”,雖然聽著是惡心了點,但意思就是這樣的,那么
當執行緒接收到這個“話語”時,它必須要做出反應,自己讓自己停止,當然,執行緒也可以根據自己的需要,選擇不停止而繼續執行

這才是和執行緒互動最安全的方式,就像一個高速行駛的汽車,只有自己慢慢停下來才是最好的方式,直接通過外力干預,很大概率是車毀人亡,

這種柔和的處理方式,在計算機里有個專用名詞,叫
中斷這是一種互動方式,你對別人發送一個中斷,別人要回應這個中斷并做出相應的處理,如果別人不回應你的這個中斷,那只能是“熱臉貼冷屁股”,完全沒了面子可見,參與中斷的雙方必須要提前約定好,你怎么發送,別人怎么處理,否則只能是雞同鴨講

Thread類和中斷相關的方法有三個:

實體方法,
void interrupt(),表示中斷執行緒,要中斷哪個執行緒就在哪個執行緒的物件上呼叫該方法,

 

Thread t = new Thread(() -> {doSomething();});t.start();t.interrupt();

 

new一個執行緒,啟動它,然后中斷它

當一個執行緒被其它執行緒中斷后,這個執行緒必須要能檢測到自己被中斷了才行,于是就有了下面這個方法,

實體方法,
boolean isInterrupted(),回傳一個執行緒是否被中斷,常用于一個執行緒檢測自己是否被中斷,

 

if(Thread.currentThread().isInterrupted()) {  doSomething();  return;}

 

如果執行緒發現自己被中斷,做一些事情,然后退出,該方法只會去讀取執行緒的中斷狀態,而不會去修改它,所以多次呼叫回傳同樣的結果

執行緒在處理中斷前,需要將中斷狀態清除一下,即將它設定成false,否則下次檢測時還是true,以為又中斷了呢,實則不是,

靜態方法,
static boolean interrupted(),該方法有兩個作用,一是回傳執行緒是否被中斷,二是如果中斷則清除中斷狀態

 

Thread.interrupted();

 

由于這個方法是靜態方法,所以只能用于當前執行緒,即執行緒自己清除自己的中斷狀態

由于
這個方法會清除中斷狀態,所以,如果第一次呼叫回傳true的話,緊接著再呼叫一次應該回傳false,除非在兩次呼叫之間執行緒真的又被中斷了

還有
一種特殊情況就是,在你中斷一個執行緒時,這個執行緒恰巧沒有在運行,它可能是因為競爭物件的監視器“失敗”(即沒有爭取上)而處于阻塞狀態,可能是因為條件不滿足而處于等待狀態,可能是因為在睡眠中,總之,執行緒目前沒有在執行代碼

由于
執行緒目前沒有在執行代碼,所以根本就無法去檢測這個中斷狀態,也就是無法回應中斷了,這樣肯定是不行的,所以設計者們此時選擇了拋例外

因此,
不管是由于阻塞/等待/睡眠,只要一個執行緒處于“停止”(即沒有在運行)時,此時去中斷它,執行緒會被喚醒,接著同樣要去再次獲取監視器,然后就收到了InterruptedException例外了,我們可以捕獲這個例外并處理它,使執行緒可以繼續正常運行,此時既然已經收到例外了,所以中斷狀態也就同時給清除了,因為中斷例外已經足夠表示中斷了

仔細想想
這種設計其實頗具人性化,就好比一個人,在他醒著的時候,跟他說話,他一定會回應你,當他睡著時,跟他說話,其實他是聽不到的,自然無法回應你,此時應該采取稍微暴力一點的手段,比如把他搖晃醒

所以,
一個執行緒正在運行時,去中斷它,是不會拋例外的,只是設定中斷狀態,此時中斷狀態就表示了中斷,一個執行緒在沒有運行時(阻塞/等待/睡眠),去中斷它,會拋出中斷例外,同時清除中斷狀態,此時中斷例外就表示了中斷

然后就是
sleep方法,表示執行緒臨時停止執行一段時間,這里只有一個要點,就是在睡眠期間,執行緒擁有的所有物件的監視器都不會被釋放

 

Thread.sleep(1000);

 

由于sleep是靜態方法,所以,一個執行緒只能讓自己睡眠,而沒有辦法讓別的執行緒睡眠,這是完全正確的,符合我們一直在闡述的思想,一個執行緒的行為應該由自己掌控,別的執行緒頂多可以給你一個中斷而已,而且你還可以選擇處理它或忽略它

最后一個方法是
join,它是一個實體方法,所以需要在一個執行緒物件上呼叫它,如下:

 

Thread t = new Thread(() -> {doSomething();});t.start();t.join();

 

表示當前執行緒執行完t.join()代碼后,就會進入等待,直到執行緒t死亡后,當前執行緒才會重新恢復執行,我在上一篇文章中把它比喻為插隊,執行緒t插到了當前執行緒的前面,所以必須等執行緒t執行完后,當前執行緒才會接著執行

這里主要是想說下它的
原始碼實作join方法標有synchronized關鍵字,所以是同步方法,而且在方法體內呼叫了從Object類繼承來的wait方法

所以join方法可以這樣來解釋,
當前執行緒獲取到執行緒物件t的監視器,然后執行t.wait(),使當前執行緒在執行緒物件t上等待,當前執行緒從運行狀態進入到等待狀態,由于物件t是一個執行緒,這是非常特殊的,因為執行緒執行完是會終止的,且在終止時會自動呼叫notifyAll方法進行通知

有句話是這樣講的,
“鳥之將死,其鳴也哀;人之將死,其言也善”,因此,一個執行緒都快要死了,是不是應該通知在自己身上等待的其它所有執行緒,把大伙都喚醒,總不能讓所有人都給自己“陪葬”吧,哈哈,

因此,在執行緒t執行結束后,會自動執行t.notifyAll()來通知所有在t上等待的執行緒,并把它們全部喚醒,所以當前執行緒會繼續接著執行,

為什么說notifyAll()是自動執行的呢?因為原始碼中并沒有去呼叫它,而實際卻執行了,所以只能是系統自動呼叫了

所以,
從宏觀上看,就是當前執行緒在等待執行緒t的死亡

任何Java物件都有監視器,所以執行緒物件也有監視器,但執行緒物件確實比較特殊,所以它的wait/notify方法也會有特殊的地方,因此官方建議我們不要隨意去玩Thread類的這些方法

 

完整示例原始碼:

https://github.com/coding-new-talking/java-code-demo.git

 

如果以上內容閣下全部都知道,而且理解到位,那已經很厲害了,請等待下篇多線的文章吧,

 

 

(END)

 

作者是作業超過10年的碼農,現在任架構師,喜歡研究技術,崇尚簡單快樂,追求以通俗易懂的語言解說技術,希望所有的讀者都能看懂并記住,下面是公眾號的二維碼,歡迎關注!

 

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/176782.html

標籤:Java

上一篇:被纏上了,小王問我怎么在 Spring Boot 中使用 JDBC 連接 MySQL

下一篇:Spring Boot Sample 001之spring-boot-begin-hello

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more