如何使用Go中的Weighted實現資源管理

来源:https://www.cnblogs.com/chenjiazhan/archive/2023/05/31/17447348.html
-Advertisement-
Play Games

> 本文首發於公眾號:Hunter後端 > 原文鏈接:[Python連接es筆記四之創建和刪除操作](https://mp.weixin.qq.com/s/ZCe0JT9TDEiZI7M5dxC9qA) 這一篇筆記介紹一下索引和數據的創建和刪除。 其實對於索引來說,如果可以接觸到 kibana 的話 ...


1. 簡介

本文將介紹 Go 語言中的 Weighted 併發原語,包括 Weighted 的基本使用方法、實現原理、使用註意事項等內容。能夠更好地理解和應用 Weighted 來實現資源的管理,從而提高程式的穩定性。

2. 問題引入

在微服務架構中,我們的服務節點負責接收其他節點的請求,並提供相應的功能和數據。比如賬戶服務,其他服務需要獲取賬戶信息,都會通過rpc請求向賬戶服務發起請求。

這些服務節點通常以集群的方式部署在伺服器上,用於處理大量的併發請求。每個伺服器都有其處理能力的上限,超過該上限可能導致性能下降甚至崩潰。

在部署服務時,通常會評估服務的併發量,併為其分配適當的資源以處理預期的請求負載。然而,在微服務架構中,存在著上游服務請求下游服務的場景。如果上游服務在某些情況下沒有正確考慮併發量,或者由於某些異常情況導致大量請求發送給下游服務,那麼下游服務可能面臨超過其處理能力的問題。這可能導致下游服務的響應時間增加,甚至無法正常處理請求,進而影響整個系統的穩定性和可用性。下麵用一個簡單的代碼來說明一下:

package main

import (
        "fmt"
        "net/http"
        "sync"
)

func main() {
        // 啟動下游服務,用於處理請求
        http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
                // 模擬下游服務的處理邏輯
                // ...
                // 完成請求處理後,從等待組中刪除一個等待
                wg.Done()
        })
        // 啟動下游服務的 HTTP 伺服器
        http.ListenAndServe(":8080", nil)

}

這裡啟動一個簡單的HTTP伺服器,由其來模擬下游服務,來接收上游服務的請求。下麵我們啟動一個簡單的程式,由其來模擬上游服務發送請求:

func main() {
        // 創建一個等待組,用於等待所有請求完成
        var wg sync.WaitGroup
        // 模擬上游服務發送大量請求給下游服務
        go func() {
                for i := 0; i < 1000000; i++ {
                        wg.Add(1)
                        go sendRequest(&wg)
                }
        }()
        // 等待所有請求完成
        wg.Wait()
}

func sendRequest(wg *sync.WaitGroup) {
        // 模擬上游服務發送請求給下游服務
        resp, err := http.Get("http://localhost:8080/")
        if err != nil {
                fmt.Println("請求失敗:", err)
        } else {
                fmt.Println("請求成功:", resp.Status)
        }

        // 請求完成後,通知等待組
        wg.Done()
}

這裡,我們同時啟動了1000000個協程同時往HTTP伺服器發送請求,如果伺服器配置不夠高,亦或者是請求量更多的情況下,已經超過了伺服器的處理上限,伺服器沒有主夠的資源去處理這些請求,此時將有可能直接將伺服器打掛掉,服務直接不可用。在這種情況下,如果由於上游服務的問題,導致下游服務,甚至整個鏈路的系統都直接崩潰,這個是不合理的,此時需要有一些手段保護下游服務由於異常流量導致整個系統的崩潰。

這裡對上面的場景進行分析,可以發現,此時是由於上游服務大量請求的過來,而當前服務並沒有足夠的資源去處理這些請求,但是並沒有對其加以限制,而是繼續處理,最終導致了整個系統的不可用。那麼此時就應該進行限流,對併發請求量進行控制,對伺服器能夠處理的併發數進行合理評估,當併發請求數超過了限制,此時應該直接拒絕其訪問,避免整個系統的不可用。

那問題來了,go語言中,有什麼方法能夠實現資源的管理,如果沒有足夠的資源,此時將直接返回,不對請求進行處理呢?其實go語言中有Weighted類型,在這種場景還挺合適的。下麵我們將對其進行介紹。

3. 基本使用

3.1 基本介紹

Weighted 是 Go 語言中 golang.org/x/sync包中的一種類型,用於限制併發訪問某個資源的數量。它提供了一種機制,允許調用者以不同的權重請求訪問資源,併在資源可用時進行授予。

Weighted的定義如下,提供了Acquire,TryAcquire,Release三個方法:

type Weighted struct {
   size    int64
   cur     int64
   mu      sync.Mutex
   waiters list.List
}
func (s *Weighted) Acquire(ctx context.Context, n int64) error{}
func (s *Weighted) TryAcquire(n int64) bool{}
func (s *Weighted) Release(n int64) {}
 
  • Acquire: 以權重 n 請求獲取資源,阻塞直到資源可用或上下文 ctx 結束。
  • TryAcquire: 嘗試以權重 n 獲取信號量,如果成功則返回 true,否則返回 false,並保持信號量不變。
  • Release:釋放具有權重 n 的信號量。

3.2 權重說明

有時候,不同請求對資源的消耗是不同的。通過設置權重,你可以更好地控制不同請求對資源的使用情況。例如,某些請求可能需要更多的計算資源或更長的處理時間,你可以設置較高的權重來確保它們能夠獲取到足夠的資源。

其次就是權重大隻是代表著請求需要使用到的資源多,對於優先順序並不會有作用。在Weighted 中,資源的許可是以先進先出(FIFO)的順序分配的,而不是根據權重來決定獲取的優先順序。當有多個請求同時等待獲取資源時,它們會按照先後順序依次獲取資源的許可。

假設先請求權重為 1 的資源,然後再請求權重為 2 的資源。如果當前可用的資源許可足夠滿足兩個請求的總權重,那麼先請求的權重為 1 的資源會先獲取到許可,然後是後續請求的權重為 2 的資源。

w.Acquire(context.Background(), 1) // 權重為 1 的請求先獲取到資源許可
w.Acquire(context.Background(), 2) // 權重為 2 的請求在權重為 1 的請求之後獲取到資源許可

3.3 基本使用

當使用Weighted來控制資源的併發訪問時,通常需要以下幾個步驟:

  • 創建Weighted實例,定義好最大資源數
  • 當需要資源時,調用Acquire方法占據資源
  • 當處理完成之後,調用Release方法釋放資源

下麵是一個簡單的代碼的示例,展示瞭如何使用Weighted實現資源控制:

func main() {
   // 1. 創建一個信號量實例,設置最大併發數
   sem := semaphore.NewWeighted(10)

   // 具體處理請求的函數
   handleRequest := func(id int) {
      // 2. 調用Acquire嘗試獲取資源
      err := sem.Acquire(context.Background(), 1)
      if err != nil {
         fmt.Printf("Goroutine %d failed to acquire resource\n", id)
      }
      // 3. 成功獲取資源,使用defer,在任務執行完之後,自動釋放資源
      defer sem.Release(1)
      // 執行業務邏輯
      return
   }

   // 模擬併發請求
   for i := 0; i < 20; i++ {
      go handleRequest(i)
   }

   time.Sleep(20 * time.Second)
}

首先,調用NewWeighted方法創建一個信號量實例,設置最大併發數為10。然後在每次請求處理前調用Acquire方法嘗試獲取資源,成功獲取資源後,使用defer關鍵字,在任務執行完後自動釋放資源,調用Release方法釋放一個資源。

保證最多同時有10個協程獲取資源。如果有更多的協程嘗試獲取資源,它們會等待其他協程釋放資源後再進行獲取。

4. 實現原理

4.1 設計初衷

Weighted類型的設計初衷是為了在併發環境中實現對資源的控制和限制。它提供了一種簡單而有效的機制,允許在同一時間內只有一定數量的併發操作可以訪問或使用特定的資源。

4.2 基本原理

Weighted類型的基本實現原理是基於計數信號量的概念。計數信號量是一種用於控制併發訪問的同步原語,它維護一個可用資源的計數器。在Weighted中,該計數器表示可用的資源數量。

當一個任務需要獲取資源時,它會調用Acquire方法。該方法首先會檢查當前可用資源的數量,如果大於零,則表示有可用資源,並將計數器減一,任務獲取到資源,並繼續執行。如果當前可用資源的數量為零,則任務會被阻塞,直到有其他任務釋放資源。

當一個任務完成對資源的使用後,它會調用Release方法來釋放資源。該方法會將計數器加一,表示資源已經可用,其他被阻塞的任務可以繼續獲取資源並執行。

通過這種方式,Weighted實現了對資源的限制和控制。它確保在同一時間內只有一定數量的併發任務可以訪問資源,超過限制的任務會被阻塞,直到有其他任務釋放資源。這樣可以有效地避免資源過度使用和競爭,保證系統的穩定性和性能。

4.3 代碼實現

4.3.1 結構體定義

Weighted的結構體定義如下:

type Weighted struct {
   size    int64
   cur     int64
   mu      sync.Mutex
   waiters list.List
}
  • size:表示資源的總數量,即可以同時獲取的最大資源數量。
  • cur:表示當前已經被獲取的資源數量。
  • mu:用於保護Weighted類型的互斥鎖,確保併發安全性。
  • waiters:使用雙向鏈表來存儲等待獲取資源的任務。

4.3.2 Acquire方法

Acquire方法將獲取指定數量的資源。如果當前可用資源數量不足,調用此方法的任務將被阻塞,並加入到等待隊列中。

func (s *Weighted) Acquire(ctx context.Context, n int64) error {
   // 1. 使用互斥鎖s.mu對Weighted類型進行加鎖,確保併發安全性。
   s.mu.Lock()
   // size - cur 代表剩餘可用資源數,如果大於請求資源數n, 此時代表剩餘可用資源 大於 需要的資源數
   // 其次,Weighted資源分配的順序是FIFO,如果等待隊列不為空,當前請求就需要自動放到隊列最後面
   if s.size-s.cur >= n && s.waiters.Len() == 0 {
      s.cur += n
      s.mu.Unlock()
      return nil
   }
    // s.size 代表最大資源數,如果需要的資源數 大於 最大資源數,此時直接返回錯誤
   if n > s.size {
      // Don't make other Acquire calls block on one that's doomed to fail.
      s.mu.Unlock()
      <-ctx.Done()
      return ctx.Err()
   }
   // 這裡代表著當前暫時獲取不到資源,此時將創建一個waiter對象放到等待隊列最後
   ready := make(chan struct{})
   // waiter對象中包含需要獲取的資源數量n和通知通道ready。
   w := waiter{n: n, ready: ready}
   // 將waiter對象放到隊列最後
   elem := s.waiters.PushBack(w)
   // 釋放鎖,讓其他請求進來
   s.mu.Unlock()

   select {
   // 如果ctx.Done()通道被關閉,表示上下文已取消,任務需要返回錯誤。
   case <-ctx.Done():
      err := ctx.Err()
      // 新獲取鎖,檢查是否已經成功獲取資源。如果成功獲取資源,將錯誤置為nil,表示獲取成功;
      s.mu.Lock()
      select {
      // 通過判斷ready channel是否接收到信號,從而來判斷是否成功獲取資源
      case <-ready:
         err = nil
      default:
         // 判斷是否是等待隊列中第一個元素
         isFront := s.waiters.Front() == elem
         // 將該請求從等待隊列中移除
         s.waiters.Remove(elem)
         // 如果是第一個等待對象,同時還有剩餘資源,喚醒後面的waiter。說不定後面的waiter剛好符合條件
         if isFront && s.size > s.cur {
            s.notifyWaiters()
         }
      }
      s.mu.Unlock()
      return err
   // ready通道接收到數據,代表此時已經成功占據到資源了
   case <-ready:
      return nil
   }
}

Weighted對象用來控制可用資源的數量。它有兩個重要的欄位,cur和size,分別表示當前可用的資源數量和總共可用的資源數量。

當一個請求通過Acquire方法請求資源時,首先會檢查剩餘資源數量是否足夠,並且等待隊列中沒有其他請求在等待資源。如果滿足這兩個條件,請求就可以成功獲取到資源。

如果剩餘資源數量不足以滿足請求,那麼一個waiter的對象會被創建並放入等待隊列中。waiter對象包含了請求需要的資源數量n和一個用於通知的通道ready。當其他請求調用Release方法釋放資源時,它們會檢查等待隊列中的waiter對象是否滿足資源需求,如果滿足,就會將資源分配給該waiter對象,並通過ready通道來通知它可以執行業務邏輯了。

即使剩餘資源數量大於請求所需數量,如果等待隊列中存在等待的請求,新的請求也會被放入等待隊列中,而不管資源是否足夠。這可能導致一些請求長時間等待資源,導致資源的浪費和延遲。因此,在使用Weighted進行資源控制時,需要謹慎評估資源配額,並避免資源饑餓的情況發生,以免影響系統的性能和響應能力。

4.3.3 Release方法

Release方法將釋放指定數量的資源。當資源被釋放時,會檢查等待隊列中的任務。它從隊頭開始逐個檢查等待的元素,並嘗試為它們分配資源,直到最後一個不滿足資源條件的元素為止。

func (s *Weighted) Release(n int64) {
   // 1. 使用互斥鎖s.mu對Weighted類型進行加鎖,確保併發安全性。
   s.mu.Lock()
   // 2. 釋放資源
   s.cur -= n
   // 3. 異常情況處理
   if s.cur < 0 {
      s.mu.Unlock()
      panic("semaphore: released more than held")
   }
   // 4. 喚醒等待任務
   s.notifyWaiters()
   s.mu.Unlock()
}

可以看到,Release方法實現相對比較簡單,釋放資源後,便直接調用notifyWaiters方法喚醒處於等待狀態的任務。下麵來看看notifyWaiters方法的具體實現:

func (s *Weighted) notifyWaiters() {
   for {
      // 獲取隊頭元素
      next := s.waiters.Front()
      // 已經沒有處於等待狀態的協程,此時直接返回
      if next == nil {
         break // No more waiters blocked.
      }

      w := next.Value.(waiter)
      // 如果資源不滿足要求 當前waiter的要求,此時直接返回
      if s.size-s.cur < w.n {
         break
      }
      // 否則占據waiter需要的資源數
      s.cur += w.n
      // 移除等待元素
      s.waiters.Remove(next)
      // 喚醒處於等待狀態的任務,Acquire方法會 <- ready 來等待信號的到來
      close(w.ready)
   }
}

notifyWaiters方法會從隊頭開始獲取元素,判斷當前資源的剩餘數,是否滿足waiter的要求,如果滿足的話,此時先占據該waiter需要的資源,之後再將其從等待隊列中移除,最後調用close方法,喚醒處於等待狀態的任務。 之後,再繼續隊列中取出元素,判斷是否滿足條件,迴圈反覆,直到不滿足waiter的條件為止。

4.3.4 TryAcquire方法

TryAcquire方法將嘗試獲取指定數量的資源,但不會阻塞。如果可用資源不足,它會立即返回一個錯誤,而不是阻塞等待。實現比較簡單,只是簡單檢查當前資源數是否滿足要求而已,具體如下:

func (s *Weighted) TryAcquire(n int64) bool {
   s.mu.Lock()
   success := s.size-s.cur >= n && s.waiters.Len() == 0
   if success {
      s.cur += n
   }
   s.mu.Unlock()
   return success
}

5. 註意事項

5.1 及時釋放資源

當使用Weighted來管理資源時,確保在使用完資源後,及時調用Release方法釋放資源。如果不這樣做,將會導致資源泄漏,最終導致所有的請求都將無法被處理。下麵展示一個簡單的代碼說明:

package main

import (
        "fmt"
        "sync"
        "time"

        "golang.org/x/sync/semaphore"
)

func main() {
        sem := semaphore.NewWeighted(5) // 創建一個最大併發數為5的Weighted實例
        // 模擬使用資源的任務
        task := func(id int) {
                //1. 成功獲取資源
                if err := sem.Acquire(context.Background(), 1); err != nil {
                        fmt.Printf("Task %d failed to acquire resource: %s\n", id, err)
                        return
                }
                // 2. 任務處理完成之後,資源沒有被釋放
                // defer sem.Release(1) // 使用defer確保在任務完成後釋放資源
               
        }

        // 啟動多個任務併發執行
        var wg sync.WaitGroup
        for i := 0; i < 10; i++ {
                wg.Add(1)
                go func(id int) {
                        defer wg.Done()
                        task(id)
                }(i)
        }
        wg.Wait() // 等待所有任務完成
}

在上面的代碼中,我們使用Weighted來控制最大併發數為5。我們在任務中沒有調用sem.Release(1)釋放資源,這些資源將一直被占用,後面啟動的5個任務將永遠無法獲取到資源,此時將永遠不會繼續執行下去。因此,務必在使用完資源後及時調用Release方法釋放資源,以確保資源的正確回收和釋放,保證系統的穩定性和性能。

而且這裡最好使用defer語句來實現資源的釋放,避免Release函數在某些異常場景下無法被執行到。

5.2 合理設置併發數

Weighted只是提供了一種管理資源的手段,具體的併發數還需要開發人員自行根據系統的實際需求和資源限制,合理設置Weighted實例的最大併發數。過大的併發數可能導致資源過度競爭,而過小的併發數可能限制了系統的吞吐量。

具體操作可以到線上預發佈環境,不斷調整觀察,獲取到一個最合適的併發數。

5.3 考慮Weighted是否適用於當前場景

Weighted 類型可以用於限制併發訪問資源的數量,但它也存在一些潛在的缺點,需要根據具體的應用場景和需求權衡利弊。

首先是記憶體開銷,Weighted 類型使用一個 sync.Mutex 以及一個 list.List 來管理等待隊列,這可能會占用一定的記憶體開銷。對於大規模的併發處理,特別是在限制極高的情況下,可能會影響系統的記憶體消耗。

其次是Weighted 類型一旦初始化,最大併發數是固定的,無法在運行時動態調整。如果你的應用程式需要根據負載情況動態調整併發限制,可能需要使用其他機制或實現。

而且Weighted是嚴格按照FIFO請求順序來分配資源的,當某些請求的權重過大時,可能會導致其他請求饑餓,即長時間等待資源。

最後,則是由於 Weighted 類型使用了互斥鎖來保護共用狀態,因此在高併發情況下,爭奪鎖可能成為性能瓶頸,影響系統的吞吐量。

因此,在使用 Weighted 類型時,需要根據具體的應用場景和需求權衡利弊,從而來決定是否使用Weighted來實現資源的管理控制。

6. 總結

本文介紹了一種解決系統中資源管理問題的解決方案Weighted。本文從問題引出,詳細介紹了Weighted的特點和使用方法。通過瞭解Weighted的設計初衷和實現原理,讀者可以更好地理解其工作原理。

同時,文章提供了使用Weighted時需要註意的事項,如及時釋放資源、合理設置併發數等,從而幫助讀者避免潛在的問題,以及能夠在比較合適的場景下使用到Weighted類型實現資源管理。基於此,我們完成了對Weighted的介紹,希望對你有所幫助。你的點贊和收藏將是我最大的動力,比心~


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

-Advertisement-
Play Games
更多相關文章
  • > 隨著人工智慧技術的不斷發展,阿裡體育等IT大廠,推出的“樂動力”、“天天跳繩”AI運動APP,讓**雲上運動會、線上運動會、健身打卡、AI體育指導**等概念空前火熱。那麼,能否將這些在APP成功應用的場景搬上小程式,分享這些概念的紅利呢?本系列文章就帶您一步一步從零開始開發一個AI運動小程式,本 ...
  • > 隨著人工智慧技術的不斷發展,阿裡體育等IT大廠,推出的“樂動力”、“天天跳繩”AI運動APP,讓**雲上運動會、線上運動會、健身打卡、AI體育指導**等概念空前火熱。那麼,能否將這些在APP成功應用的場景搬上小程式,分享這些概念的紅利呢?本系列文章就帶您一步一步從零開始開發一個AI運動小程式,本 ...
  • # 前言 本文主要講述**單例模式**,文中使用通俗易懂的案例,使你更好的學習本章知識點並理解原理,做到有道無術。 # 一. 什麼是單例模式 單例模式是23種設計模式中**創建型模式**的一種,通過單例模式的方法創建的類在當前進程或者線程中只有一個實例。單例模式有兩種比較常見的實現方式:**餓漢式* ...
  • ### 外觀式定義 為子系統中的一組介面提供一個一致的界面,Facade 模式定義了一個高層介面,這個介面使得這一子系統更加容易使用。 #### 界面 在這裡提到的界面,主要指的是從一個組件外部來看這個組件,能夠看到什麼,這就是這個組件的界面,也就是所說的外觀。 #### 介面 在這裡提到的介面,主 ...
  • # 前言 本文主要講述**工廠模式**,文中使用通俗易懂的案例,使你更好的學習本章知識點並理解原理,做到有道無術。 # 一.什麼是工廠模式 工廠模式是23種設計模式中**創建型模式**的一種,它是一個最簡單的對象創建管理方式,根據調用方傳遞的類型來創建對象並返回。封裝了對象創建的過程,降低了程式模塊 ...
  • ### 解釋器模式(Interpreter Pattern) #### 一、定義 解釋器模式(Interpreter Pattern)提供了評估語言的語法或表達式的方式,它屬於行為型模式。這種模式實現了一個表達式介面,該介面解釋一個特定的上下文。這種模式被用在 SQL 解析、符號處理引擎等。 給定一 ...
  • 在Java中,序列化(Serialization)是指將對象的狀態轉換為位元組流的過程,以便將其保存到文件、在網路中傳輸或持久化到資料庫中。而反序列化(Deserialization)則是將位元組流轉換回對象的過程,恢復對象的狀態。 序列化和反序列化主要用於以下場景: 1. 對象持久化:通過序列化,可以 ...
  • > 內容摘自我的學習網站:topjavaer.cn Redis連環40問,絕對夠全! ## Redis是什麼? Redis(`Remote Dictionary Server`)是一個使用 C 語言編寫的,高性能非關係型的鍵值對資料庫。與傳統資料庫不同的是,Redis 的數據是存在記憶體中的,所以讀寫 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...