fix: 修复深度克隆时 map/slice 中基本类型值被错误存储为指针的问题
- 提取 cloneSequence 函数统一处理 slice/array 克隆 - 提取 cloneElement 函数处理单个元素克隆 - 提取 needsDeepClone 函数判断是否需要深度克隆 - 减少代码重复,提高可维护性 为 ExampleNew_withValue* 系列测试添加 New() 后的输出, 清晰展示深度克隆后的初始状态和值类型正确性。
This commit is contained in:
parent
8fbc859e74
commit
f9e7f8e781
@ -71,6 +71,8 @@ func ExampleNew_withValue() {
|
|||||||
|
|
||||||
// 传入值 - 会创建深度克隆,修改不会影响原始数据
|
// 传入值 - 会创建深度克隆,修改不会影响原始数据
|
||||||
rfx := New(person)
|
rfx := New(person)
|
||||||
|
fmt.Printf("After New: %+v\n", rfx.Any())
|
||||||
|
|
||||||
rfx.Set("Name", "Bob")
|
rfx.Set("Name", "Bob")
|
||||||
rfx.Set("Age", 35)
|
rfx.Set("Age", 35)
|
||||||
|
|
||||||
@ -83,6 +85,7 @@ func ExampleNew_withValue() {
|
|||||||
rfx.Get("Age").Int())
|
rfx.Get("Age").Int())
|
||||||
|
|
||||||
// Output:
|
// Output:
|
||||||
|
// After New: {Name:Alice Age:30}
|
||||||
// Original - Name: Alice, Age: 30
|
// Original - Name: Alice, Age: 30
|
||||||
// Clone - Name: Bob, Age: 35
|
// Clone - Name: Bob, Age: 35
|
||||||
}
|
}
|
||||||
@ -97,6 +100,8 @@ func ExampleNew_withValue_map() {
|
|||||||
|
|
||||||
// 传入 map 值 - 会创建深度克隆
|
// 传入 map 值 - 会创建深度克隆
|
||||||
rfx := New(config)
|
rfx := New(config)
|
||||||
|
fmt.Printf("After New: %v\n", rfx.Any())
|
||||||
|
|
||||||
rfx.Set("host", "127.0.0.1")
|
rfx.Set("host", "127.0.0.1")
|
||||||
rfx.Set("port", 9090)
|
rfx.Set("port", 9090)
|
||||||
|
|
||||||
@ -109,6 +114,7 @@ func ExampleNew_withValue_map() {
|
|||||||
rfx.Get("port").Int())
|
rfx.Get("port").Int())
|
||||||
|
|
||||||
// Output:
|
// Output:
|
||||||
|
// After New: map[host:localhost port:8080]
|
||||||
// Original - Host: localhost, Port: 8080
|
// Original - Host: localhost, Port: 8080
|
||||||
// Clone - Host: 127.0.0.1, Port: 9090
|
// Clone - Host: 127.0.0.1, Port: 9090
|
||||||
}
|
}
|
||||||
@ -120,6 +126,8 @@ func ExampleNew_withValue_slice() {
|
|||||||
|
|
||||||
// 传入 slice 值 - 会创建深度克隆
|
// 传入 slice 值 - 会创建深度克隆
|
||||||
rfx := New(items)
|
rfx := New(items)
|
||||||
|
fmt.Printf("After New: %v\n", rfx.Any())
|
||||||
|
|
||||||
rfx.Set("0", "orange")
|
rfx.Set("0", "orange")
|
||||||
rfx.Set("2", "grape")
|
rfx.Set("2", "grape")
|
||||||
|
|
||||||
@ -133,6 +141,7 @@ func ExampleNew_withValue_slice() {
|
|||||||
rfx.Get("2").String())
|
rfx.Get("2").String())
|
||||||
|
|
||||||
// Output:
|
// Output:
|
||||||
|
// After New: [apple banana cherry]
|
||||||
// Original: [apple banana cherry]
|
// Original: [apple banana cherry]
|
||||||
// Clone: [orange banana grape]
|
// Clone: [orange banana grape]
|
||||||
}
|
}
|
||||||
@ -158,6 +167,8 @@ func ExampleNew_withValue_nestedStruct() {
|
|||||||
|
|
||||||
// 传入值 - 会创建深度克隆,包括嵌套的结构体
|
// 传入值 - 会创建深度克隆,包括嵌套的结构体
|
||||||
rfx := New(person)
|
rfx := New(person)
|
||||||
|
fmt.Printf("After New: %+v\n", rfx.Any())
|
||||||
|
|
||||||
rfx.Set("Name", "Bob")
|
rfx.Set("Name", "Bob")
|
||||||
rfx.Set("Address.City", "Shanghai")
|
rfx.Set("Address.City", "Shanghai")
|
||||||
|
|
||||||
@ -170,6 +181,7 @@ func ExampleNew_withValue_nestedStruct() {
|
|||||||
rfx.Get("Address", "City").String())
|
rfx.Get("Address", "City").String())
|
||||||
|
|
||||||
// Output:
|
// Output:
|
||||||
|
// After New: {Name:Alice Address:{City:Beijing Street:Main St}}
|
||||||
// Original - Name: Alice, City: Beijing
|
// Original - Name: Alice, City: Beijing
|
||||||
// Clone - Name: Bob, City: Shanghai
|
// Clone - Name: Bob, City: Shanghai
|
||||||
}
|
}
|
||||||
|
|||||||
55
util.go
55
util.go
@ -61,14 +61,10 @@ func CloneValue(dst, src reflect.Value) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
dst.Set(reflect.MakeSlice(src.Type(), src.Len(), src.Cap()))
|
dst.Set(reflect.MakeSlice(src.Type(), src.Len(), src.Cap()))
|
||||||
for i := 0; i < src.Len(); i++ {
|
cloneSequence(dst, src)
|
||||||
CloneValue(dst.Index(i), src.Index(i))
|
|
||||||
}
|
|
||||||
|
|
||||||
case reflect.Array:
|
case reflect.Array:
|
||||||
for i := 0; i < src.Len(); i++ {
|
cloneSequence(dst, src)
|
||||||
CloneValue(dst.Index(i), src.Index(i))
|
|
||||||
}
|
|
||||||
|
|
||||||
case reflect.Map:
|
case reflect.Map:
|
||||||
if src.IsNil() {
|
if src.IsNil() {
|
||||||
@ -77,10 +73,8 @@ func CloneValue(dst, src reflect.Value) {
|
|||||||
dst.Set(reflect.MakeMap(src.Type()))
|
dst.Set(reflect.MakeMap(src.Type()))
|
||||||
for _, key := range src.MapKeys() {
|
for _, key := range src.MapKeys() {
|
||||||
val := src.MapIndex(key)
|
val := src.MapIndex(key)
|
||||||
// Map 的值需要特殊处理
|
clonedVal := cloneElement(val)
|
||||||
newVal := reflect.New(val.Type()).Elem()
|
dst.SetMapIndex(key, clonedVal)
|
||||||
CloneValue(newVal, val)
|
|
||||||
dst.SetMapIndex(key, newVal)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -91,6 +85,47 @@ func CloneValue(dst, src reflect.Value) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// cloneSequence 克隆序列类型(slice/array)的元素
|
||||||
|
func cloneSequence(dst, src reflect.Value) {
|
||||||
|
for i := 0; i < src.Len(); i++ {
|
||||||
|
elem := src.Index(i)
|
||||||
|
clonedElem := cloneElement(elem)
|
||||||
|
dst.Index(i).Set(clonedElem)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// cloneElement 克隆单个元素,处理 interface 包装的情况
|
||||||
|
// 对于复杂类型(struct, map, slice, array)进行深度克隆
|
||||||
|
// 对于基本类型,直接返回原始值
|
||||||
|
func cloneElement(elem reflect.Value) reflect.Value {
|
||||||
|
// 解引用以获取实际值
|
||||||
|
actualElem := elem
|
||||||
|
for actualElem.Kind() == reflect.Interface || actualElem.Kind() == reflect.Ptr {
|
||||||
|
if actualElem.IsNil() {
|
||||||
|
return elem
|
||||||
|
}
|
||||||
|
actualElem = actualElem.Elem()
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果元素需要递归克隆(struct, map, slice, array)
|
||||||
|
if actualElem.IsValid() && needsDeepClone(actualElem.Kind()) {
|
||||||
|
clonedElem := reflect.New(actualElem.Type()).Elem()
|
||||||
|
CloneValue(clonedElem, actualElem)
|
||||||
|
return clonedElem
|
||||||
|
}
|
||||||
|
|
||||||
|
// 对于基本类型,直接返回原始值
|
||||||
|
return elem
|
||||||
|
}
|
||||||
|
|
||||||
|
// needsDeepClone 判断某个类型是否需要深度克隆
|
||||||
|
func needsDeepClone(kind reflect.Kind) bool {
|
||||||
|
return kind == reflect.Struct ||
|
||||||
|
kind == reflect.Map ||
|
||||||
|
kind == reflect.Slice ||
|
||||||
|
kind == reflect.Array
|
||||||
|
}
|
||||||
|
|
||||||
// expandPath 展开路径,支持点号分割
|
// expandPath 展开路径,支持点号分割
|
||||||
// 例如: expandPath("a.b", "c") -> []string{"a", "b", "c"}
|
// 例如: expandPath("a.b", "c") -> []string{"a", "b", "c"}
|
||||||
func expandPath(p ...string) []string {
|
func expandPath(p ...string) []string {
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user