1
0
mirror of https://github.com/gogf/gf.git synced 2025-04-05 03:05:05 +08:00
gf/util/gconv/internal/structcache/structcache_cached.go

119 lines
4.0 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 structcache
import (
"reflect"
"github.com/gogf/gf/v2/internal/utils"
"github.com/gogf/gf/v2/util/gtag"
)
// GetCachedStructInfo retrieves or parses and returns a cached info for certain struct type.
// The given `structType` should be type of struct.
func (cf *Converter) GetCachedStructInfo(structType reflect.Type, priorityTag string) *CachedStructInfo {
if structType.Kind() != reflect.Struct {
return nil
}
// check if it has been cached.
cachedStructInfo, ok := cf.getCachedConvertStructInfo(structType)
if ok {
// directly returns the cached struct info if already exists.
return cachedStructInfo
}
// else create one.
// it parses and generates a cache info for given struct type.
cachedStructInfo = NewCachedStructInfo(cf)
var (
priorityTagArray []string
parentIndex = make([]int, 0)
)
if priorityTag != "" {
priorityTagArray = append(utils.SplitAndTrim(priorityTag, ","), gtag.StructTagPriority...)
} else {
priorityTagArray = gtag.StructTagPriority
}
cf.parseStructToCachedStructInfo(structType, parentIndex, cachedStructInfo, priorityTagArray)
cf.storeCachedStructInfo(structType, cachedStructInfo)
return cachedStructInfo
}
func (cf *Converter) storeCachedStructInfo(structType reflect.Type, cachedStructInfo *CachedStructInfo) {
// Temporarily enabled as an experimental feature
cf.cachedStructsInfoMap.Store(structType, cachedStructInfo)
}
func (cf *Converter) getCachedConvertStructInfo(structType reflect.Type) (*CachedStructInfo, bool) {
// Temporarily enabled as an experimental feature
v, ok := cf.cachedStructsInfoMap.Load(structType)
if ok {
return v.(*CachedStructInfo), ok
}
return nil, false
}
// parseStructToCachedStructInfo parses given struct reflection type and stores its fields info into given CachedStructInfo.
// It stores nothing into CachedStructInfo if given struct reflection type has no fields.
func (cf *Converter) parseStructToCachedStructInfo(
structType reflect.Type,
fieldIndexes []int,
cachedStructInfo *CachedStructInfo,
priorityTagArray []string,
) {
var (
fieldName string
structField reflect.StructField
fieldType reflect.Type
)
// TODO:
// Check if the structure has already been cached in the cache.
// If it has been cached, some information can be reused,
// but the [FieldIndex] needs to be reset.
// We will not implement it temporarily because it is somewhat complex
for i := 0; i < structType.NumField(); i++ {
structField = structType.Field(i)
fieldType = structField.Type
fieldName = structField.Name
// Only do converting to public attributes.
if !utils.IsLetterUpper(fieldName[0]) {
continue
}
copyFieldIndexes := make([]int, len(fieldIndexes))
copy(copyFieldIndexes, fieldIndexes)
// normal basic attributes.
if structField.Anonymous {
// handle struct attributes, it might be struct/*struct embedded..
if fieldType.Kind() == reflect.Ptr {
fieldType = fieldType.Elem()
}
if fieldType.Kind() != reflect.Struct {
continue
}
// Skip the embedded structure of the 0 field,
if fieldType.NumField() == 0 {
continue
}
if structField.Tag != "" {
// Do not add anonymous structures without tags
cachedStructInfo.AddField(structField, append(copyFieldIndexes, i), priorityTagArray)
}
cf.parseStructToCachedStructInfo(fieldType, append(copyFieldIndexes, i), cachedStructInfo, priorityTagArray)
continue
}
// Do not directly use append(fieldIndexes, i)
// When the structure is nested deeply, it may lead to bugs,
// which are caused by the slice expansion mechanism
// So it is necessary to allocate a separate index for each field
// See details https://github.com/gogf/gf/issues/3789
cachedStructInfo.AddField(structField, append(copyFieldIndexes, i), priorityTagArray)
}
}