1
0
mirror of https://github.com/gogf/gf.git synced 2025-04-05 03:05:05 +08:00

fix(util/gconv): fix incompatable converting to nil pointer from older version implement

This commit is contained in:
John Guo 2025-03-25 20:27:02 +08:00
parent 69e3362d0d
commit eb407045c7
2 changed files with 42 additions and 9 deletions

View File

@ -805,3 +805,38 @@ func Test_Issue3903(t *testing.T) {
t.Assert(a.UserId, 100)
})
}
// https://github.com/gogf/gf/issues/4218
func Test_Issue4218(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
type SysMenuVo struct {
MenuId int64 `json:"menuId" orm:"menu_id"`
MenuName string `json:"menuName" orm:"menu_name"`
Children []*SysMenuVo `json:"children" orm:"children"`
ParentId int64 `json:"parentId" orm:"parent_id"`
}
menus := []*SysMenuVo{
{
MenuId: 1,
MenuName: "系统管理",
ParentId: 0,
},
{
MenuId: 2,
MenuName: "字典查询",
ParentId: 1,
},
}
var parent *SysMenuVo
err := gconv.Scan(menus[0], &parent)
t.AssertNil(err)
t.Assert(parent.MenuId, 1)
t.Assert(parent.ParentId, 0)
parent.Children = append(parent.Children, menus[1])
t.Assert(len(menus[0].Children), 1)
t.Assert(menus[0].Children[0].MenuId, 2)
t.Assert(menus[0].Children[0].ParentId, 1)
})
}

View File

@ -87,11 +87,14 @@ func (c *Converter) Scan(srcValue any, dstPointer any, option ...ScanOption) (er
}
// Get the element type and kind of dstPointer
var (
dstPointerReflectValueElem = dstPointerReflectValue.Elem()
dstPointerReflectValueElemKind = dstPointerReflectValueElem.Kind()
)
var dstPointerReflectValueElem = dstPointerReflectValue.Elem()
// Check if srcValue and dstPointer are the same type, in which case direct assignment can be performed
if ok := c.doConvertWithTypeCheck(srcValueReflectValue, dstPointerReflectValueElem); ok {
return nil
}
// Handle multiple level pointers
var dstPointerReflectValueElemKind = dstPointerReflectValueElem.Kind()
if dstPointerReflectValueElemKind == reflect.Ptr {
if dstPointerReflectValueElem.IsNil() {
// Create a new value for the pointer dereference
@ -105,11 +108,6 @@ func (c *Converter) Scan(srcValue any, dstPointer any, option ...ScanOption) (er
return c.Scan(srcValueReflectValue, dstPointerReflectValueElem, option...)
}
// Check if srcValue and dstPointer are the same type, in which case direct assignment can be performed
if ok := c.doConvertWithTypeCheck(srcValueReflectValue, dstPointerReflectValueElem); ok {
return nil
}
scanOption := c.getScanOption(option...)
// Handle different destination types
switch dstPointerReflectValueElemKind {