首頁 > 後端開發 > Golang > 主體

如何在 Golang 泛型中用具體類型填入介面片段?

Barbara Streisand
發布: 2024-10-26 05:38:30
原創
371 人瀏覽過

How to Fill a Slice of Interfaces with Concrete Types in Golang Generics?

Golang 泛型:同時使用介面和具體類型

在 Go 1.18 中,泛型為類型處理引入了新的可能性。然而,當一起使用介面和具體類型時,某些場景可能會帶來挑戰。

嘗試建立這樣的函數時會出現這樣的場景:

<code class="go">func Fill[X any](slice []*X){
   for i := range slice {
      slice[i] = new(X)
   }
}</code>
登入後複製

此函數旨在填充具有具體類型的介面片段。例如,可以使用此函數以 new(int) 填充 *int 陣列。

嘗試使用實作介面的具體類型填充介面切片時會出現問題。考慮以下程式碼:

<code class="go">func Fill[X, Y any](slice []X){
   for i := range slice {
      slice[i] = new(Y) // not work!
   }
}

xs := make([]sync.Locker, 10) // fill with nils
Fill[sync.Locker,sync.Mutex](xs) // ouch</code>
登入後複製

在這種情況下,函數不起作用,因為將 X 和 Y 都約束為任何會破壞介面和實作類型之間的關係。編譯器在編譯時僅辨識 X 和 Y 是不同的型別。

可能的解

有一個解法可以使用明確斷言來編譯程式碼:

<code class="go">func Fill[X, Y any](slice []X) {
    for i := range slice {
        slice[i] = any(*new(Y)).(X)
    }
}</code>
登入後複製

但是,這個解有一個顯著的缺點:如果Y 沒有實作X,它會發生恐慌,就像使用sync.Locker 和sync.Mutex 一樣。此外,對 Y 使用指標類型會導致 nil 值,因為它會遺失基本類型和零值資訊。

更好的替代方案

更強大的解決方案是使用構造函數而不是第二個類型參數:

<code class="go">func main() {
    xs := make([]sync.Locker, 10)
    Fill(xs, func() sync.Locker { return &sync.Mutex{} })
}

func Fill[X any](slice []X, f func() X) {
    for i := range slice {
        slice[i] = f()
    }
}</code>
登入後複製

在這種方法中,函數採用X 類型的切片和建構函數。建構函數會建立一個 X 類型的實例,用具體實例填滿切片。

以上是如何在 Golang 泛型中用具體類型填入介面片段?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
作者最新文章
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板
關於我們 免責聲明 Sitemap
PHP中文網:公益線上PHP培訓,幫助PHP學習者快速成長!