You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ntool/nreflect/convert.go

198 lines
5.0 KiB
Go

package nreflect
import (
"fmt"
"git.noahlan.cn/noahlan/ntool/internal/convert"
"git.noahlan.cn/noahlan/ntool/ndef"
"git.noahlan.cn/noahlan/ntool/nmath"
"git.noahlan.cn/noahlan/ntool/nstr"
"reflect"
"strconv"
)
// BaseTypeVal convert custom type or intX,uintX,floatX to generic base type.
//
// intX/unitX => int64
// floatX => float64
// string => string
//
// returns int64,string,float or error
func BaseTypeVal(v reflect.Value) (value any, err error) {
v = reflect.Indirect(v)
switch v.Kind() {
case reflect.String:
value = v.String()
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
value = v.Int()
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
value = int64(v.Uint()) // always return int64
case reflect.Float32, reflect.Float64:
value = v.Float()
default:
err = ndef.ErrConvType
}
return
}
// ValueByType create reflect.Value by give reflect.Type
func ValueByType(val any, typ reflect.Type) (rv reflect.Value, err error) {
// handle kind: string, bool, intX, uintX, floatX
if typ.Kind() == reflect.String || typ.Kind() <= reflect.Float64 {
return ValueByKind(val, typ.Kind())
}
newRv := reflect.ValueOf(val)
// try auto convert slice type
if IsArrayOrSlice(newRv.Kind()) && IsArrayOrSlice(typ.Kind()) {
return ConvSlice(newRv, typ.Elem())
}
// check type. like map
if newRv.Type() == typ {
return newRv, nil
}
err = ndef.ErrConvType
return
}
// ValueByKind create reflect.Value by give reflect.Kind
//
// TIPs:
//
// Only support kind: string, bool, intX, uintX, floatX
func ValueByKind(val any, kind reflect.Kind) (rv reflect.Value, err error) {
switch kind {
case reflect.Int:
if dstV, err1 := nmath.ToInt(val); err1 == nil {
rv = reflect.ValueOf(dstV)
}
case reflect.Int8:
if dstV, err1 := nmath.ToInt(val); err1 == nil {
rv = reflect.ValueOf(int8(dstV))
}
case reflect.Int16:
if dstV, err1 := nmath.ToInt(val); err1 == nil {
rv = reflect.ValueOf(int16(dstV))
}
case reflect.Int32:
if dstV, err1 := nmath.ToInt(val); err1 == nil {
rv = reflect.ValueOf(int32(dstV))
}
case reflect.Int64:
if dstV, err1 := nmath.ToInt64(val); err1 == nil {
rv = reflect.ValueOf(dstV)
}
case reflect.Uint:
if dstV, err1 := nmath.ToUint(val); err1 == nil {
rv = reflect.ValueOf(uint(dstV))
}
case reflect.Uint8:
if dstV, err1 := nmath.ToUint(val); err1 == nil {
rv = reflect.ValueOf(uint8(dstV))
}
case reflect.Uint16:
if dstV, err1 := nmath.ToUint(val); err1 == nil {
rv = reflect.ValueOf(uint16(dstV))
}
case reflect.Uint32:
if dstV, err1 := nmath.ToUint(val); err1 == nil {
rv = reflect.ValueOf(uint32(dstV))
}
case reflect.Uint64:
if dstV, err1 := nmath.ToUint(val); err1 == nil {
rv = reflect.ValueOf(dstV)
}
case reflect.Float32:
if dstV, err1 := nmath.ToFloat(val); err1 == nil {
rv = reflect.ValueOf(float32(dstV))
}
case reflect.Float64:
if dstV, err1 := nmath.ToFloat(val); err1 == nil {
rv = reflect.ValueOf(dstV)
}
case reflect.String:
if dstV, err1 := nstr.ToString(val); err1 == nil {
rv = reflect.ValueOf(dstV)
}
case reflect.Bool:
if bl, err := convert.ToBool(val); err == nil {
rv = reflect.ValueOf(bl)
}
}
if !rv.IsValid() {
err = ndef.ErrConvType
}
return
}
// ConvSlice make new type slice from old slice, will auto convert element type.
//
// TIPs:
//
// Only support kind: string, bool, intX, uintX, floatX
func ConvSlice(oldSlRv reflect.Value, newElemTyp reflect.Type) (rv reflect.Value, err error) {
if !IsArrayOrSlice(oldSlRv.Kind()) {
panic("only allow array or slice type value")
}
// do not need convert type
if oldSlRv.Type().Elem() == newElemTyp {
return oldSlRv, nil
}
newSlTyp := reflect.SliceOf(newElemTyp)
newSlRv := reflect.MakeSlice(newSlTyp, 0, 0)
for i := 0; i < oldSlRv.Len(); i++ {
newElemV, err := ValueByKind(oldSlRv.Index(i).Interface(), newElemTyp.Kind())
if err != nil {
return reflect.Value{}, err
}
newSlRv = reflect.Append(newSlRv, newElemV)
}
return newSlRv, nil
}
// String convert
func String(rv reflect.Value) string {
s, _ := ValToString(rv, false)
return s
}
// ToString convert
func ToString(rv reflect.Value) (str string, err error) {
return ValToString(rv, true)
}
// ValToString convert handle
func ValToString(rv reflect.Value, defaultAsErr bool) (str string, err error) {
rv = Indirect(rv)
switch rv.Kind() {
case reflect.Invalid:
str = ""
case reflect.Bool:
str = strconv.FormatBool(rv.Bool())
case reflect.String:
str = rv.String()
case reflect.Float32:
str = strconv.FormatFloat(rv.Float(), 'f', -1, 32)
case reflect.Float64:
str = strconv.FormatFloat(rv.Float(), 'f', -1, 64)
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
str = strconv.FormatInt(rv.Int(), 10)
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
str = strconv.FormatUint(rv.Uint(), 10)
default:
if defaultAsErr {
err = ndef.ErrConvType
} else {
str = fmt.Sprint(rv.Interface())
}
}
return
}