gf/internal/empty/empty.go

178 lines
3.7 KiB
Go
Raw Normal View History

// Copyright 2019 gf Author(https://github.com/gogf/gf). 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.
2020-05-17 18:16:26 +08:00
// Package empty provides functions for checking empty variables.
package empty
import (
2019-06-05 18:40:26 +08:00
"reflect"
)
// apiString is used for type assert api for String().
type apiString interface {
String() string
}
// apiInterfaces is used for type assert api for Interfaces.
type apiInterfaces interface {
Interfaces() []interface{}
}
// apiMapStrAny is the interface support for converting struct parameter to map.
type apiMapStrAny interface {
MapStrAny() map[string]interface{}
}
2019-06-13 21:14:46 +08:00
// IsEmpty checks whether given <value> empty.
2020-04-13 23:44:43 +08:00
// It returns true if <value> is in: 0, nil, false, "", len(slice/map/chan) == 0,
// or else it returns false.
func IsEmpty(value interface{}) bool {
2019-06-19 09:06:52 +08:00
if value == nil {
return true
}
// It firstly checks the variable as common types using assertion to enhance the performance,
// and then using reflection.
2019-06-19 09:06:52 +08:00
switch value := value.(type) {
case int:
return value == 0
case int8:
return value == 0
case int16:
return value == 0
case int32:
return value == 0
case int64:
return value == 0
case uint:
return value == 0
case uint8:
return value == 0
case uint16:
return value == 0
case uint32:
return value == 0
case uint64:
return value == 0
case float32:
return value == 0
case float64:
return value == 0
case bool:
return value == false
case string:
return value == ""
case []byte:
return len(value) == 0
2020-04-13 23:44:43 +08:00
case []rune:
return len(value) == 0
case []int:
return len(value) == 0
case []string:
return len(value) == 0
case []float32:
return len(value) == 0
case []float64:
return len(value) == 0
case map[string]interface{}:
return len(value) == 0
2019-06-19 09:06:52 +08:00
default:
// Common interfaces checks.
if f, ok := value.(apiString); ok {
2020-10-20 14:07:01 +08:00
if f == nil {
return true
}
return f.String() == ""
}
if f, ok := value.(apiInterfaces); ok {
2020-10-20 14:07:01 +08:00
if f == nil {
return true
}
return len(f.Interfaces()) == 0
}
if f, ok := value.(apiMapStrAny); ok {
2020-10-20 14:07:01 +08:00
if f == nil {
return true
}
return len(f.MapStrAny()) == 0
}
2019-06-19 09:06:52 +08:00
// Finally using reflect.
var rv reflect.Value
if v, ok := value.(reflect.Value); ok {
rv = v
} else {
rv = reflect.ValueOf(value)
}
2020-12-12 21:57:07 +08:00
2019-06-19 09:06:52 +08:00
switch rv.Kind() {
2020-12-12 21:57:07 +08:00
case reflect.Bool:
return !rv.Bool()
case reflect.Int,
reflect.Int8,
reflect.Int16,
reflect.Int32,
reflect.Int64:
return rv.Int() == 0
case reflect.Uint,
reflect.Uint8,
reflect.Uint16,
reflect.Uint32,
reflect.Uint64,
reflect.Uintptr:
return rv.Uint() == 0
case reflect.Float32,
reflect.Float64:
return rv.Float() == 0
case reflect.String:
return rv.Len() == 0
case reflect.Struct:
for i := 0; i < rv.NumField(); i++ {
if !IsEmpty(rv) {
return false
}
}
return true
2019-06-19 09:06:52 +08:00
case reflect.Chan,
reflect.Map,
reflect.Slice,
reflect.Array:
return rv.Len() == 0
case reflect.Func,
reflect.Ptr,
reflect.Interface,
reflect.UnsafePointer:
if rv.IsNil() {
return true
}
}
}
return false
}
// IsNil checks whether given <value> is nil.
2020-02-23 20:25:55 +08:00
// Note that it might use reflect feature which affects performance a little bit.
func IsNil(value interface{}) bool {
if value == nil {
return true
}
var rv reflect.Value
if v, ok := value.(reflect.Value); ok {
rv = v
} else {
rv = reflect.ValueOf(value)
}
switch rv.Kind() {
case reflect.Chan,
reflect.Map,
reflect.Slice,
reflect.Func,
reflect.Ptr,
reflect.Interface,
reflect.UnsafePointer:
return rv.IsNil()
}
return false
}