mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 03:05:05 +08:00
158 lines
3.3 KiB
Go
158 lines
3.3 KiB
Go
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
|
//
|
|
// This Source Code Form is subject to the terms of the MIT License.
|
|
// If a copy of the MIT was not distributed with this file,
|
|
// You can obtain one at https://github.com/gogf/gf.
|
|
|
|
package converter
|
|
|
|
import (
|
|
"math"
|
|
"reflect"
|
|
"strconv"
|
|
|
|
"github.com/gogf/gf/v2/encoding/gbinary"
|
|
"github.com/gogf/gf/v2/errors/gcode"
|
|
"github.com/gogf/gf/v2/errors/gerror"
|
|
"github.com/gogf/gf/v2/internal/empty"
|
|
"github.com/gogf/gf/v2/util/gconv/internal/localinterface"
|
|
)
|
|
|
|
// Int converts `any` to int.
|
|
func (c *Converter) Int(any any) (int, error) {
|
|
if v, ok := any.(int); ok {
|
|
return v, nil
|
|
}
|
|
v, err := c.Int64(any)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return int(v), nil
|
|
}
|
|
|
|
// Int8 converts `any` to int8.
|
|
func (c *Converter) Int8(any any) (int8, error) {
|
|
if v, ok := any.(int8); ok {
|
|
return v, nil
|
|
}
|
|
v, err := c.Int64(any)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return int8(v), nil
|
|
}
|
|
|
|
// Int16 converts `any` to int16.
|
|
func (c *Converter) Int16(any any) (int16, error) {
|
|
if v, ok := any.(int16); ok {
|
|
return v, nil
|
|
}
|
|
v, err := c.Int64(any)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return int16(v), nil
|
|
}
|
|
|
|
// Int32 converts `any` to int32.
|
|
func (c *Converter) Int32(any any) (int32, error) {
|
|
if v, ok := any.(int32); ok {
|
|
return v, nil
|
|
}
|
|
v, err := c.Int64(any)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return int32(v), nil
|
|
}
|
|
|
|
// Int64 converts `any` to int64.
|
|
func (c *Converter) Int64(any any) (int64, error) {
|
|
if empty.IsNil(any) {
|
|
return 0, nil
|
|
}
|
|
if v, ok := any.(int64); ok {
|
|
return v, nil
|
|
}
|
|
rv := reflect.ValueOf(any)
|
|
switch rv.Kind() {
|
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
|
return rv.Int(), nil
|
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
|
return int64(rv.Uint()), nil
|
|
case reflect.Uintptr:
|
|
return int64(rv.Uint()), nil
|
|
case reflect.Float32, reflect.Float64:
|
|
return int64(rv.Float()), nil
|
|
case reflect.Bool:
|
|
if rv.Bool() {
|
|
return 1, nil
|
|
}
|
|
return 0, nil
|
|
case reflect.Ptr:
|
|
if rv.IsNil() {
|
|
return 0, nil
|
|
}
|
|
if f, ok := any.(localinterface.IInt64); ok {
|
|
return f.Int64(), nil
|
|
}
|
|
return c.Int64(rv.Elem().Interface())
|
|
case reflect.Slice:
|
|
// TODO: It might panic here for these types.
|
|
if rv.Type().Elem().Kind() == reflect.Uint8 {
|
|
return gbinary.DecodeToInt64(rv.Bytes()), nil
|
|
}
|
|
case reflect.String:
|
|
var (
|
|
s = rv.String()
|
|
isMinus = false
|
|
)
|
|
if len(s) > 0 {
|
|
if s[0] == '-' {
|
|
isMinus = true
|
|
s = s[1:]
|
|
} else if s[0] == '+' {
|
|
s = s[1:]
|
|
}
|
|
}
|
|
// Hexadecimal.
|
|
if len(s) > 2 && s[0] == '0' && (s[1] == 'x' || s[1] == 'X') {
|
|
if v, e := strconv.ParseInt(s[2:], 16, 64); e == nil {
|
|
if isMinus {
|
|
return -v, nil
|
|
}
|
|
return v, nil
|
|
}
|
|
}
|
|
// Decimal.
|
|
if v, e := strconv.ParseInt(s, 10, 64); e == nil {
|
|
if isMinus {
|
|
return -v, nil
|
|
}
|
|
return v, nil
|
|
}
|
|
// Float64.
|
|
valueInt64, err := c.Float64(s)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
if math.IsNaN(valueInt64) {
|
|
return 0, nil
|
|
} else {
|
|
if isMinus {
|
|
return -int64(valueInt64), nil
|
|
}
|
|
return int64(valueInt64), nil
|
|
}
|
|
default:
|
|
if f, ok := any.(localinterface.IInt64); ok {
|
|
return f.Int64(), nil
|
|
}
|
|
}
|
|
return 0, gerror.NewCodef(
|
|
gcode.CodeInvalidParameter,
|
|
`unsupport value type for converting to int64: %v`,
|
|
reflect.TypeOf(any),
|
|
)
|
|
}
|