Java併發系列[5]----ReentrantLock源碼分析

来源:https://www.cnblogs.com/liuyun1995/archive/2018/02/24/8462088.html
-Advertisement-
Play Games

在Java5.0之前,協調對共用對象的訪問可以使用的機制只有synchronized和volatile。我們知道synchronized關鍵字實現了內置鎖,而volatile關鍵字保證了多線程的記憶體可見性。在大多數情況下,這些機制都能很好地完成工作,但卻無法實現一些更高級的功能,例如,無法中斷一個正 ...


在Java5.0之前,協調對共用對象的訪問可以使用的機制只有synchronized和volatile。我們知道synchronized關鍵字實現了內置鎖,而volatile關鍵字保證了多線程的記憶體可見性。在大多數情況下,這些機制都能很好地完成工作,但卻無法實現一些更高級的功能,例如,無法中斷一個正在等待獲取鎖的線程,無法實現限定時間的獲取鎖機制,無法實現非阻塞結構的加鎖規則等。而這些更靈活的加鎖機制通常都能夠提供更好的活躍性或性能。因此,在Java5.0中增加了一種新的機制:ReentrantLock。ReentrantLock類實現了Lock介面,並提供了與synchronized相同的互斥性和記憶體可見性,它的底層是通過AQS來實現多線程同步的。與內置鎖相比ReentrantLock不僅提供了更豐富的加鎖機制,而且在性能上也不遜色於內置鎖(在以前的版本中甚至優於內置鎖)。說了ReentrantLock這麼多的優點,那麼下麵我們就來揭開它的源碼看看它的具體實現。

1.synchronized關鍵字的介紹

Java提供了內置鎖來支持多線程的同步,JVM根據synchronized關鍵字來標識同步代碼塊,當線程進入同步代碼塊時會自動獲取鎖,退出同步代碼塊時會自動釋放鎖,一個線程獲得鎖後其他線程將會被阻塞。每個Java對象都可以用做一個實現同步的鎖,synchronized關鍵字可以用來修飾對象方法,靜態方法和代碼塊,當修飾對象方法和靜態方法時鎖分別是方法所在的對象和Class對象,當修飾代碼塊時需提供額外的對象作為鎖。每個Java對象之所以可以作為鎖,是因為在對象頭中關聯了一個monitor對象(管程),線程進入同步代碼塊時會自動持有monitor對象,退出時會自動釋放monitor對象,當monitor對象被持有時其他線程將會被阻塞。當然這些同步操作都由JVM底層幫你實現了,但以synchronized關鍵字修飾的方法和代碼塊在底層實現上還是有些區別的。synchronized關鍵字修飾的方法是隱式同步的,即無需通過位元組碼指令來控制的,JVM可以根據方法表中的ACC_SYNCHRONIZED訪問標誌來區分一個方法是否是同步方法;而synchronized關鍵字修飾的代碼塊是顯式同步的,它是通過monitorenter和monitorexit位元組碼指令來控制線程對管程的持有和釋放。monitor對象內部持有_count欄位,_count等於0表示管程未被持有,_count大於0表示管程已被持有,每次持有線程重入時_count都會加1,每次持有線程退出時_count都會減1,這就是內置鎖重入性的實現原理。另外,monitor對象內部還有兩條隊列_EntryList和_WaitSet,對應著AQS的同步隊列和條件隊列,當線程獲取鎖失敗時會到_EntryList中阻塞,當調用鎖對象的wait方法時線程將會進入_WaitSet中等待,這是內置鎖的線程同步和條件等待的實現原理。

2.ReentrantLock和Synchronized的比較

synchronized關鍵字是Java提供的內置鎖機制,其同步操作由底層JVM實現,而ReentrantLock是java.util.concurrent包提供的顯式鎖,其同步操作由AQS同步器提供支持。ReentrantLock在加鎖和記憶體上提供的語義與內置鎖相同,此外它還提供了一些其他功能,包括定時的鎖等待,可中斷的鎖等待,公平鎖,以及實現非塊結構的加鎖。另外,在早期的JDK版本中ReentrantLock在性能上還占有一定的優勢,既然ReentrantLock擁有這麼多優勢,為什麼還要使用synchronized關鍵字呢?事實上確實有許多人使用ReentrantLock來替代synchronized關鍵字的加鎖操作。但是內置鎖仍然有它特有的優勢,內置鎖為許多開發人員所熟悉,使用方式也更加的簡潔緊湊,因為顯式鎖必須手動在finally塊中調用unlock,所以使用內置鎖相對來說會更加安全些。同時未來更加可能會去提升synchronized而不是ReentrantLock的性能。因為synchronized是JVM的內置屬性,它能執行一些優化,例如對線程封閉的鎖對象的鎖消除優化,通過增加鎖的粒度來消除內置鎖的同步,而如果通過基於類庫的鎖來實現這些功能,則可能性不大。所以當需要一些高級功能時才應該使用ReentrantLock,這些功能包括:可定時的,可輪詢的與可中斷的鎖獲取操作,公平隊列,以及非塊結構的鎖。否則,還是應該優先使用synchronized。

3.獲取鎖和釋放鎖的操作

我們首先來看一下使用ReentrantLock加鎖的示例代碼。

 1 public void doSomething() {
 2     //預設是獲取一個非公平鎖
 3     ReentrantLock lock = new ReentrantLock();
 4     try{
 5         //執行前先加鎖
 6         lock.lock();   
 7         //執行操作...
 8     }finally{
 9         //最後釋放鎖
10         lock.unlock();
11     }
12 }

以下是獲取鎖和釋放鎖這兩個操作的API。

1 //獲取鎖的操作
2 public void lock() {
3     sync.lock();
4 }
5 //釋放鎖的操作
6 public void unlock() {
7     sync.release(1);
8 }

可以看到獲取鎖和釋放鎖的操作分別委托給Sync對象的lock方法和release方法。

 1 public class ReentrantLock implements Lock, java.io.Serializable {
 2     
 3     private final Sync sync;
 4 
 5     abstract static class Sync extends AbstractQueuedSynchronizer {
 6         abstract void lock();
 7     }
 8     
 9     //實現非公平鎖的同步器
10     static final class NonfairSync extends Sync {
11         final void lock() {
12             ...
13         }
14     }
15     
16     //實現公平鎖的同步器
17     static final class FairSync extends Sync {
18         final void lock() {
19             ...
20         }
21     }
22 }

每個ReentrantLock對象都持有一個Sync類型的引用,這個Sync類是一個抽象內部類它繼承自AbstractQueuedSynchronizer,它裡面的lock方法是一個抽象方法。ReentrantLock的成員變數sync是在構造時賦值的,下麵我們看看ReentrantLock的兩個構造方法都做了些什麼?

1 //預設無參構造器
2 public ReentrantLock() {
3     sync = new NonfairSync();
4 }
5 
6 //有參構造器
7 public ReentrantLock(boolean fair) {
8     sync = fair ? new FairSync() : new NonfairSync();
9 }

調用預設無參構造器會將NonfairSync實例賦值給sync,此時鎖是非公平鎖。有參構造器允許通過參數來指定是將FairSync實例還是NonfairSync實例賦值給sync。NonfairSync和FairSync都是繼承自Sync類並重寫了lock()方法,所以公平鎖和非公平鎖在獲取鎖的方式上有些區別,這個我們下麵會講到。再來看看釋放鎖的操作,每次調用unlock()方法都只是去執行sync.release(1)操作,這步操作會調用AbstractQueuedSynchronizer類的release()方法,我們再來回顧一下。

 1 //釋放鎖的操作(獨占模式)
 2 public final boolean release(int arg) {
 3     //撥動密碼鎖, 看看是否能夠開鎖
 4     if (tryRelease(arg)) {
 5         //獲取head結點
 6         Node h = head;
 7         //如果head結點不為空並且等待狀態不等於0就去喚醒後繼結點
 8         if (h != null && h.waitStatus != 0) {
 9             //喚醒後繼結點
10             unparkSuccessor(h);
11         }
12         return true;
13     }
14     return false;
15 }

這個release方法是AQS提供的釋放鎖操作的API,它首先會去調用tryRelease方法去嘗試獲取鎖,tryRelease方法是抽象方法,它的實現邏輯在子類Sync裡面。

 1 //嘗試釋放鎖
 2 protected final boolean tryRelease(int releases) {
 3     int c = getState() - releases;
 4     //如果持有鎖的線程不是當前線程就拋出異常
 5     if (Thread.currentThread() != getExclusiveOwnerThread()) {
 6         throw new IllegalMonitorStateException();
 7     }
 8     boolean free = false;
 9     //如果同步狀態為0則表明鎖被釋放
10     if (c == 0) {
11         //設置鎖被釋放的標誌為真
12         free = true;
13         //設置占用線程為空
14         setExclusiveOwnerThread(null);
15     }
16     setState(c);
17     return free;
18 }

這個tryRelease方法首先會獲取當前同步狀態,並將當前同步狀態減去傳入的參數值得到新的同步狀態,然後判斷新的同步狀態是否等於0,如果等於0則表明當前鎖被釋放,然後先將鎖的釋放狀態置為真,再將當前占有鎖的線程清空,最後調用setState方法設置新的同步狀態並返回鎖的釋放狀態。

4.公平鎖和非公平鎖

我們知道ReentrantLock是公平鎖還是非公平鎖是基於sync指向的是哪個具體實例。在構造時會為成員變數sync賦值,如果賦值為NonfairSync實例則表明是非公平鎖,如果賦值為FairSync實例則表明為公平鎖。如果是公平鎖,線程將按照它們發出請求的順序來獲得鎖,但在非公平鎖上,則允許插隊行為:當一個線程請求非公平的鎖時,如果在發出請求的同時該鎖的狀態變為可用,那麼這個線程將跳過隊列中所有等待的線程直接獲得這個鎖。下麵我們先看看非公平鎖的獲取方式。

 1 //非公平同步器
 2 static final class NonfairSync extends Sync {
 3     //實現父類的抽象獲取鎖的方法
 4     final void lock() {
 5         //使用CAS方式設置同步狀態
 6         if (compareAndSetState(0, 1)) {
 7             //如果設置成功則表明鎖沒被占用
 8             setExclusiveOwnerThread(Thread.currentThread());
 9         } else {
10             //否則表明鎖已經被占用, 調用acquire讓線程去同步隊列排隊獲取
11             acquire(1);
12         }
13     }
14     //嘗試獲取鎖的方法
15     protected final boolean tryAcquire(int acquires) {
16         return nonfairTryAcquire(acquires);
17     }
18 }
19 
20 //以不可中斷模式獲取鎖(獨占模式)
21 public final void acquire(int arg) {
22     if (!tryAcquire(arg) && acquireQueued(addWaiter(Node.EXCLUSIVE), arg)) {
23         selfInterrupt();
24     }
25 }

可以看到在非公平鎖的lock方法中,線程第一步就會以CAS方式將同步狀態的值從0改為1。其實這步操作就等於去嘗試獲取鎖,如果更改成功則表明線程剛來就獲取了鎖,而不必再去同步隊列裡面排隊了。如果更改失敗則表明線程剛來時鎖還未被釋放,所以接下來就調用acquire方法。我們知道這個acquire方法是繼承自AbstractQueuedSynchronizer的方法,現在再來回顧一下該方法,線程進入acquire方法後首先去調用tryAcquire方法嘗試去獲取鎖,由於NonfairSync覆蓋了tryAcquire方法,併在方法中調用了父類Sync的nonfairTryAcquire方法,所以這裡會調用到nonfairTryAcquire方法去嘗試獲取鎖。我們看看這個方法具體做了些什麼。

 1 //非公平的獲取鎖
 2 final boolean nonfairTryAcquire(int acquires) {
 3     //獲取當前線程
 4     final Thread current = Thread.currentThread();
 5     //獲取當前同步狀態
 6     int c = getState();
 7     //如果同步狀態為0則表明鎖沒有被占用
 8     if (c == 0) {
 9         //使用CAS更新同步狀態
10         if (compareAndSetState(0, acquires)) {
11             //設置目前占用鎖的線程
12             setExclusiveOwnerThread(current);
13             return true;
14         }
15     //否則的話就判斷持有鎖的是否是當前線程
16     }else if (current == getExclusiveOwnerThread()) {
17         //如果鎖是被當前線程持有的, 就直接修改當前同步狀態
18         int nextc = c + acquires;
19         if (nextc < 0) {
20             throw new Error("Maximum lock count exceeded");
21         }
22         setState(nextc);
23         return true;
24     }
25     //如果持有鎖的不是當前線程則返回失敗標誌
26     return false;
27 }

nonfairTryAcquire方法是Sync的方法,我們可以看到線程進入此方法後首先去獲取同步狀態,如果同步狀態為0就使用CAS操作更改同步狀態,其實這又是獲取了一遍鎖。如果同步狀態不為0表明鎖被占用,此時會先去判斷持有鎖的線程是否是當前線程,如果是的話就將同步狀態加1,否則的話這次嘗試獲取鎖的操作宣告失敗。於是會調用addWaiter方法將線程添加到同步隊列。綜上來看,在非公平鎖的模式下一個線程在進入同步隊列之前會嘗試獲取兩遍鎖,如果獲取成功則不進入同步隊列排隊,否則才進入同步隊列排隊。接下來我們看看公平鎖的獲取方式。

 1 //實現公平鎖的同步器
 2 static final class FairSync extends Sync {
 3     //實現父類的抽象獲取鎖的方法
 4     final void lock() {
 5         //調用acquire讓線程去同步隊列排隊獲取
 6         acquire(1);
 7     }
 8     //嘗試獲取鎖的方法
 9     protected final boolean tryAcquire(int acquires) {
10         //獲取當前線程
11         final Thread current = Thread.currentThread();
12         //獲取當前同步狀態
13         int c = getState();
14         //如果同步狀態0則表示鎖沒被占用
15         if (c == 0) {
16             //判斷同步隊列是否有前繼結點
17             if (!hasQueuedPredecessors() && compareAndSetState(0, acquires)) {
18                 //如果沒有前繼結點且設置同步狀態成功就表示獲取鎖成功
19                 setExclusiveOwnerThread(current);
20                 return true;
21             }
22         //否則判斷是否是當前線程持有鎖
23         }else if (current == getExclusiveOwnerThread()) {
24             //如果是當前線程持有鎖就直接修改同步狀態
25             int nextc = c + acquires;
26             if (nextc < 0) {
27                 throw new Error("Maximum lock count exceeded");
28             }
29             setState(nextc);
30             return true;
31         }
32         //如果不是當前線程持有鎖則獲取失敗
33         return false;
34     }
35 }

調用公平鎖的lock方法時會直接調用acquire方法。同樣的,acquire方法首先會調用FairSync重寫的tryAcquire方法來嘗試獲取鎖。在該方法中也是首先獲取同步狀態的值,如果同步狀態為0則表明此時鎖剛好被釋放,這時和非公平鎖不同的是它會先去調用hasQueuedPredecessors方法查詢同步隊列中是否有人在排隊,如果沒人在排隊才會去修改同步狀態的值,可以看到公平鎖在這裡採取禮讓的方式而不是自己馬上去獲取鎖。除了這一步和非公平鎖不一樣之外,其他的操作都是一樣的。綜上所述,可以看到公平鎖在進入同步隊列之前只檢查了一遍鎖的狀態,即使是發現了鎖是開的也不會自己馬上去獲取,而是先讓同步隊列中的線程先獲取,所以可以保證在公平鎖下所有線程獲取鎖的順序都是先來後到的,這也保證了獲取鎖的公平性。
那麼我們為什麼不希望所有鎖都是公平的呢?畢竟公平是一種好的行為,而不公平是一種不好的行為。由於線程的掛起和喚醒操作存在較大的開銷而影響系統性能,特別是在競爭激烈的情況下公平鎖將導致線程頻繁的掛起和喚醒操作,而非公平鎖可以減少這樣的操作,所以在性能上將會優於公平鎖。另外,由於大部分線程使用鎖的時間都是非常短暫的,而線程的喚醒操作會存在延時情況,有可能在A線程被喚醒期間B線程馬上獲取了鎖並使用完釋放了鎖,這就導致了雙贏的局面,A線程獲取鎖的時刻並沒有推遲,但B線程提前使用了鎖,並且吞吐量也獲得了提高。

5.條件隊列的實現機制

內置條件隊列存在一些缺陷,每個內置鎖都只能有一個相關聯的條件隊列,這導致多個線程可能在同一個條件隊列上等待不同的條件謂詞,那麼每次調用notifyAll時都會將所有等待的線程喚醒,當線程醒來後發現並不是自己等待的條件謂詞,轉而又會被掛起。這導致做了很多無用的線程喚醒和掛起操作,而這些操作將會大量浪費系統資源,降低系統的性能。如果想編寫一個帶有多個條件謂詞的併發對象,或者想獲得除了條件隊列可見性之外的更多控制權,就需要使用顯式的Lock和Condition而不是內置鎖和條件隊列。一個Condition和一個Lock關聯在一起,就像一個條件隊列和一個內置鎖相關聯一樣。要創建一個Condition,可以在相關聯的Lock上調用Lock.newCondition方法。我們先來看一個使用Condition的示例。

 1 public class BoundedBuffer {
 2 
 3     final Lock lock = new ReentrantLock();
 4     final Condition notFull = lock.newCondition();   //條件謂詞:notFull
 5     final Condition notEmpty = lock.newCondition();  //條件謂詞:notEmpty
 6     final Object[] items = new Object[100];
 7     int putptr, takeptr, count;
 8 
 9     //生產方法
10     public void put(Object x) throws InterruptedException {
11         lock.lock();
12         try {
13             while (count == items.length)
14                 notFull.await();  //隊列已滿, 線程在notFull隊列上等待
15             items[putptr] = x;
16             if (++putptr == items.length) putptr = 0;
17             ++count;
18             notEmpty.signal(); //生產成功, 喚醒notEmpty隊列的結點
19         } finally {
20             lock.unlock();
21         }
22     }
23 
24     //消費方法
25     public Object take() throws InterruptedException {
26         lock.lock();
27         try {
28             while (count == 0)
29                 notEmpty.await(); //隊列為空, 線程在notEmpty隊列上等待
30             Object x = items[takeptr];
31             if (++takeptr == items.length) takeptr = 0;
32             --count;
33             notFull.signal();  //消費成功, 喚醒notFull隊列的結點
34             return x;
35         } finally {
36             lock.unlock();
37         }
38     }
39     
40 }

一個lock對象可以產生多個條件隊列,這裡產生了兩個條件隊列notFull和notEmpty。當容器已滿時再調用put方法的線程需要進行阻塞,等待條件謂詞為真(容器不滿)才醒來繼續執行;當容器為空時再調用take方法的線程也需要阻塞,等待條件謂詞為真(容器不空)才醒來繼續執行。這兩類線程是根據不同的條件謂詞進行等待的,所以它們會進入兩個不同的條件隊列中阻塞,等到合適時機再通過調用Condition對象上的API進行喚醒。下麵是newCondition方法的實現代碼。

 1 //創建條件隊列
 2 public Condition newCondition() {
 3     return sync.newCondition();
 4 }
 5 
 6 abstract static class Sync extends AbstractQueuedSynchronizer {
 7     //新建Condition對象
 8     final ConditionObject newCondition() {
 9         return new ConditionObject();
10     }
11 }

ReentrantLock上的條件隊列的實現都是基於AbstractQueuedSynchronizer的,我們在調用newCondition方法時所獲得的Condition對象就是AQS的內部類ConditionObject的實例。所有對條件隊列的操作都是通過調用ConditionObject對外提供的API來完成的。有關於ConditionObject的具體實現大家可以查閱我的這篇文章《Java併發系列[4]----AbstractQueuedSynchronizer源碼分析之條件隊列》,這裡就不重覆贅述了。至此,我們對ReentrantLock源碼的剖析也告一段落,希望閱讀本篇文章能夠對讀者們理解並掌握ReentrantLock起到一定的幫助作用。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 眾所周知,在ES6之前,JavaScript是沒有塊級作用域的,如下圖所示: 學過其他語言的同學肯定有點詫異,為什麼會這樣呢?因為js還是不同於其他語言的,在ES5中,只有全局作用域和函數作用域,並沒有塊作用域,當然我們可以實現塊作用域的功能。看下麵代碼: 在這段段代碼中,我們使用立即執行函數(II ...
  • 新年好呀~過個年光打游戲,function都寫不順溜了。 上一節的代碼到這裡了: 經過長長的resolve,最終也只是解析入口文件的合法路徑信息,然後調用回調函數。 接下來分析回調函數是如何處理返回結果的: 返回的結果有兩部分,一個是loader,一個是文件對應路徑。 對於入口文件的當前解析,不存在 ...
  • 在 export defaul new Router({ )} 這個路由配置中一定要加mode : ‘history’ 否者就會在路由前面預設添加# 路由跳轉的幾種方式: ...
  • ES6 模塊的設計思想,是儘量的靜態化,使得編譯時就能確定模塊的依賴關係,以及輸入和輸出的變數。 ES6 模塊不是對象,而是通過export命令顯式指定輸出的代碼,再通過import命令輸入。 需要特別註意的是,export命令規定的是對外的介面,必須與模塊內部的變數建立一一對應關係。 export ...
  • ⚠️組件的作用域是孤立的,vue解決組件傳值問題是通過props ⚠️子傳父的時候需要vm.$emit觸發實例上的事件,頁面需要定一個方法去取值 ⚠️一定要註意命名方式和書寫,例如mylChart和myl-chart dataRadio和data-radio :objline和@data-radio ...
  • 之前的文章大量的內容在和大家探討分散式存儲,接下來的章節進入了分散式計算領域。坦白說,個人之前專業的重心側重於存儲,對許多計算的內容理解可能不是和確切,如果文章中的理解有所不妥,願虛心賜教。本篇將和大家聊一聊分散式計算的一個子集: 批處理 。 批處理系統通常也叫離線系統 ,需要大量的輸入數據,運行一 ...
  • Related Links:Zuul https://github.com/Netflix/zuulCAT https://github.com/dianping/catApollo https://github.com/ctripcorp/apolloKairosDB https://github... ...
  • 前面我們曾有篇文章中提到過關於用tensorflow訓練手寫28 28像素點的數字的識別,在那篇文章中我們把手寫數字圖像直接碾壓成了一個784列的數據進行識別,但實際上,這個圖像是28 28長寬結構的,我們這次使用CNN捲積神經網路來進行識別。 捲積神經網路我的理解是部分模仿了人眼的功能。 我們在看 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...