Golang泛型實(shí)現(xiàn)類型轉(zhuǎn)換的方法實(shí)例
1.前言
Golang 標(biāo)準(zhǔn)庫(kù)提供了很多類型轉(zhuǎn)換的函數(shù),如 strconv 包可完成 string 與基本數(shù)據(jù)類型之間的轉(zhuǎn)換。
比如將 int 與 string 之間的互轉(zhuǎn)。
// int to string s := strconv.Itoa(i) // string to int i, err := strconv.ParseInt(i, 0, 64)
如果我們想完成任意類型到某一具體類型的轉(zhuǎn)換,該如何實(shí)現(xiàn)呢?
2.To String
以 string 為,我們可以這樣實(shí)現(xiàn)。
// ToStringE casts any type to a string type. func ToStringE(i any) (string, error) { i = indirectToStringerOrError(i) switch s := i.(type) { case string: return s, nil case bool: return strconv.FormatBool(s), nil case float64: return strconv.FormatFloat(s, 'f', -1, 64), nil case float32: return strconv.FormatFloat(float64(s), 'f', -1, 32), nil case int: return strconv.Itoa(s), nil case int64: return strconv.FormatInt(s, 10), nil case int32: return strconv.Itoa(int(s)), nil case int16: return strconv.FormatInt(int64(s), 10), nil case int8: return strconv.FormatInt(int64(s), 10), nil case uint: return strconv.FormatUint(uint64(s), 10), nil case uint64: return strconv.FormatUint(uint64(s), 10), nil case uint32: return strconv.FormatUint(uint64(s), 10), nil case uint16: return strconv.FormatUint(uint64(s), 10), nil case uint8: return strconv.FormatUint(uint64(s), 10), nil case json.Number: return s.String(), nil case []byte: return string(s), nil case template.HTML: return string(s), nil case template.URL: return string(s), nil case template.JS: return string(s), nil case template.CSS: return string(s), nil case template.HTMLAttr: return string(s), nil case nil: return "", nil case fmt.Stringer: return s.String(), nil case error: return s.Error(), nil default: return "", fmt.Errorf("unable to cast %#v of type %T to string", i, i) } }
其中 indirectToStringerOrError 是對(duì)指針類型的解引用,從標(biāo)準(zhǔn)庫(kù) html/template/content.go 獲取。
var ( errorType = reflect.TypeOf((*error)(nil)).Elem() fmtStringerType = reflect.TypeOf((*fmt.Stringer)(nil)).Elem() ) // Copied from html/template/content.go. // indirectToStringerOrError returns the value, after dereferencing as many times // as necessary to reach the base type (or nil) or an implementation of fmt.Stringer // or error, func indirectToStringerOrError(a any) any { if a == nil { return nil } v := reflect.ValueOf(a) for !v.Type().Implements(fmtStringerType) && !v.Type().Implements(errorType) && v.Kind() == reflect.Pointer && !v.IsNil() { v = v.Elem() } return v.Interface() }
3.To Other Type
那么對(duì)其他類型我們也都要實(shí)現(xiàn)對(duì)應(yīng)的轉(zhuǎn)換函數(shù)。
// ToBoolE casts any type to a bool type. func ToBoolE(i any) (bool, error) { i = indirect(i) switch b := i.(type) { case bool: return b, nil case nil: return false, nil case int: if i.(int) != 0 { return true, nil } return false, nil case string: return strconv.ParseBool(i.(string)) default: return false, fmt.Errorf("unable to cast %#v of type %T to bool", i, i) } } // ToIntE, ToInt8E, ToInt16E...
3.泛型
最終,我們可以通過(guò)泛型完成對(duì)上面多個(gè)具體類型轉(zhuǎn)換函數(shù)的封裝。這樣我們只需要調(diào)用一個(gè)函數(shù),便可完成對(duì)所有類型的轉(zhuǎn)換。
// ToAnyE converts one type to another and returns an error if occurred. func ToAnyE[T any](a any) (T, error) { var t T switch any(t).(type) { case bool: v, err := ToBoolE(a) if err != nil { return t, err } t = any(v).(T) case int: v, err := ToIntE(a) if err != nil { return t, err } t = any(v).(T) case int8: v, err := ToInt8E(a) if err != nil { return t, err } t = any(v).(T) case int16: v, err := ToInt16E(a) if err != nil { return t, err } t = any(v).(T) case int32: v, err := ToInt32E(a) if err != nil { return t, err } t = any(v).(T) case int64: v, err := ToInt64E(a) if err != nil { return t, err } t = any(v).(T) case uint: v, err := ToUintE(a) if err != nil { return t, err } t = any(v).(T) case uint8: v, err := ToUint8E(a) if err != nil { return t, err } t = any(v).(T) case uint16: v, err := ToUint16E(a) if err != nil { return t, err } t = any(v).(T) case uint32: v, err := ToUint32E(a) if err != nil { return t, err } t = any(v).(T) case uint64: v, err := ToUint64E(a) if err != nil { return t, err } t = any(v).(T) case float32: v, err := ToFloat32E(a) if err != nil { return t, err } t = any(v).(T) case float64: v, err := ToFloat64E(a) if err != nil { return t, err } t = any(v).(T) case string: v, err := ToStringE(a) if err != nil { return t, err } t = any(v).(T) default: return t, fmt.Errorf("the type %T is not supported", t) } return t, nil }
如果不關(guān)心錯(cuò)誤,可以再封裝一下。
// ToAny converts one type to another type. func ToAny[T any](a any) T { v, _ := ToAnyE[T](a) return v }
4.使用示例
package main import ( "fmt" ) func main() { fmt.Println(ToAny[string](1)) // "1" fmt.Println(ToAny[string](true)) // "true" fmt.Println(ToAny[string](1.1)) // "1.1" fmt.Println(ToAny[int]("1")) // 1 fmt.Println(ToAny[int]("1.0")) // 1 fmt.Println(ToAny[int](true)) // 1 fmt.Println(ToAny[bool]("true")) // true fmt.Println(ToAny[bool]("false")) // false fmt.Println(ToAny[bool]("True")) // true fmt.Println(ToAny[bool]("False")) // false fmt.Println(ToAny[bool](1)) // true fmt.Println(ToAny[bool](0)) // false fmt.Println(ToAny[bool](nil)) // false }
5.go-huge-util
為了方便大家使用,以上相關(guān)代碼已開(kāi)源至 Github 工具庫(kù) go-huge-util,大家可使用 go mod 方式 import 然后使用。
import "github.com/dablelv/go-huge-util/conv" conv.ToAny[string](1) // "1" conv.ToAny[string](true) // "true" conv.ToAny[string](1.1) // "1.1" conv.ToAny[int]("1") // 1 conv.ToAny[int]("1.0") // 1 conv.ToAny[int](true) // 1 conv.ToAny[bool]("true") // true conv.ToAny[bool]("false") // false conv.ToAny[bool]("True") // true conv.ToAny[bool]("False") // false conv.ToAny[bool](1) // true conv.ToAny[bool](0) // false conv.ToAny[bool](nil) // false
go-huge-util 除了類型轉(zhuǎn)換,還有很多其他實(shí)用函數(shù),如加解密、zip 等,歡迎大家使用、Star 和 Pull Request。
參考文獻(xiàn)
github.com/dablelv/go-huge-util
總結(jié)
到此這篇關(guān)于Golang泛型實(shí)現(xiàn)類型轉(zhuǎn)換的文章就介紹到這了,更多相關(guān)Golang泛型類型轉(zhuǎn)換內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- 一文帶你吃透Golang中的類型轉(zhuǎn)換
- 一文帶你了解Golang中類型轉(zhuǎn)換庫(kù)cast的使用
- golang類型推斷與隱式類型轉(zhuǎn)換
- golang強(qiáng)制類型轉(zhuǎn)換和類型斷言
- Golang 語(yǔ)言極簡(jiǎn)類型轉(zhuǎn)換庫(kù)cast的使用詳解
- golang值類型轉(zhuǎn)換成[]uint8類型的操作
- golang類型轉(zhuǎn)換組件Cast的使用詳解
- golang的強(qiáng)制類型轉(zhuǎn)換實(shí)現(xiàn)
- Golang 類型轉(zhuǎn)換的實(shí)現(xiàn)(斷言、強(qiáng)制、顯式類型)
相關(guān)文章
Golang中println和fmt.Println區(qū)別解析
Golang 中打印數(shù)據(jù)通常使用 fmt.Println() 方法,也可以使用內(nèi)置的 println() 方法。這兩個(gè)方法大家可能都使用過(guò),它們的區(qū)別是什么呢?本文給大家詳細(xì)講解,感興趣的朋友跟隨小編一起看看吧2023-03-03go語(yǔ)言實(shí)現(xiàn)http服務(wù)端與客戶端的例子
今天小編就為大家分享一篇go語(yǔ)言實(shí)現(xiàn)http服務(wù)端與客戶端的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-08-08Golang監(jiān)聽(tīng)日志文件并發(fā)送到kafka中
這篇文章主要介紹了Golang監(jiān)聽(tīng)日志文件并發(fā)送到kafka中,日志收集項(xiàng)目的準(zhǔn)備中,本文主要講的是利用golang的tail庫(kù),監(jiān)聽(tīng)日志文件的變動(dòng),將日志信息發(fā)送到kafka中?,需要的朋友可以參考一下2022-04-04Golang連接Redis數(shù)據(jù)庫(kù)的方法
這篇文章主要介紹了Golang連接Redis數(shù)據(jù)庫(kù)的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-12-12