123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152 |
- // Package mhayaUtils file from https://github.com/gogf/gf
- package mhayaUtils
- import "reflect"
- // IsEmpty checks whether given <value> empty.
- // 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 {
- if value == nil {
- return true
- }
- // It firstly checks the variable as common types using assertion to enhance the performance,
- // and then using reflection.
- switch val := value.(type) {
- case int:
- return val == 0
- case int8:
- return val == 0
- case int16:
- return val == 0
- case int32:
- return val == 0
- case int64:
- return val == 0
- case uint:
- return val == 0
- case uint8:
- return val == 0
- case uint16:
- return val == 0
- case uint32:
- return val == 0
- case uint64:
- return val == 0
- case float32:
- return val == 0
- case float64:
- return val == 0
- case bool:
- return !val
- case string:
- return val == ""
- case []byte:
- return len(val) == 0
- case []rune:
- return len(val) == 0
- case []int:
- return len(val) == 0
- case []string:
- return len(val) == 0
- case []float32:
- return len(val) == 0
- case []float64:
- return len(val) == 0
- case map[string]interface{}:
- return len(val) == 0
- default:
- // Finally using reflect.
- var rv reflect.Value
- if v, ok := value.(reflect.Value); ok {
- rv = v
- } else {
- rv = reflect.ValueOf(value)
- }
- switch rv.Kind() {
- 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
- 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.
- // Parameter <traceSource> is used for tracing to the source variable if given <value> is type
- // of a pinter that also points to a pointer. It returns nil if the source is nil when <traceSource>
- // is true.
- // Note that it might use reflect feature which affects performance a little bit.
- func IsNil(value interface{}, traceSource ...bool) 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.Interface,
- reflect.UnsafePointer:
- return !rv.IsValid() || rv.IsNil()
- case reflect.Ptr:
- if len(traceSource) > 0 && traceSource[0] {
- for rv.Kind() == reflect.Ptr {
- rv = rv.Elem()
- }
- if !rv.IsValid() {
- return true
- }
- if rv.Kind() == reflect.Ptr {
- return rv.IsNil()
- }
- } else {
- return !rv.IsValid() || rv.IsNil()
- }
- }
- return false
- }
|