php小編百草為您介紹Go語言1.20版本中的重要功能——編譯時的嚴格可比性。在Go語言1.20版本中,引入了新的編譯器標志,可以確保編譯時生成的二進制文件在不同編譯環境下的可比性。這意味著,不同編譯環境下生成的二進制文件將具有相同的行為和結果,從而減少了由于編譯環境不同而引起的潛在問題。這一功能的引入將進一步提高Go語言的可靠性和穩定性,為開發者提供更好的開發體驗。
問題內容
在 Go 1.18 和 Go 1.19 中,我可以在編譯時確保類型嚴格可比,即它支持 == 和 != 運算符,并且保證這些運算符 運行時不要驚慌。
這很有用,例如可以避免無意中向結構添加字段,從而導致不必要的恐慌。
我只是嘗試用它實例化 comparable :
// supports == and != but comparison could panic at run time
type Foo struct {
SomeField any
}
func ensureComparable[T comparable]() {
// no-op
}
var _ = ensureComparable[Foo] // doesn't compile because Foo comparison may panic
登錄后復制
由于 comparable 約束的定義,這在 Go 1.18 和 1.19 中是可能的:
The predeclared interface type comparable denotes the set of all non-interface types that are comparable
登錄后復制
盡管 Go 1.18 和 1.19 規范沒有提及不是接口但也不能嚴格比較的類型,例如[2]fmt.Stringer 或 struct { foo any },gc 編譯器確實拒絕將這些作為 comparable 的參數。
有幾個示例的游樂場:https://go.dev/play/p/_Ggfdnn6OzZ
在 Go 1.20 中,實例化 comparable 將與更廣泛的可比性概念保持一致。這使得 ensureComparable[Foo] 編譯即使我不希望它。
有沒有辦法靜態確保與 Go 1.20 的嚴格可比性?
解決方法
要測試 Foo 在 Go 1.20 中是否嚴格可比,請使用受 Foo 約束的類型參數實例化 ensureComparable。
// unchanged
type Foo struct {
SomeField any
}
// unchanged
func ensureComparable[T comparable]() {}
// T constrained by Foo, instantiate ensureComparable with T
func ensureStrictlyComparable[T Foo]() {
_ = ensureComparable[T] // <---- doesn't compile
}
登錄后復制
此解決方案最初是由 Robert Griesemer 在此建議 a>.
那么它是如何工作的呢?
Go 1.20 引入了實現接口和滿足約束:
第二個要點是允許接口和帶有接口的類型實例化 comparable 的例外。
現在在 Go 1.20 中,由于可滿足性異常,類型 Foo 本身可以實例化 comparable 。但類型參數 T 不是 Foo。類型參數的兼容性定義不同:
T 的類型集包含一個不嚴格可比的類型 Foo (因為它有一個接口字段),因此 T 不滿足 comparable。即使 Foo 本身也是如此。
如果 Foo 的運算符 == 和 != 在運行時可能會出現恐慌,則此技巧有效地使程序無法編譯。






