由 go orm 引发的探索

前言

今天遇到了一个 bug, 是 golang 的orm导致的. 使用了gorm框架. 通过实现ScanValue可以将数据库中的 json 内容解析出来, 免除了 字符串再解码的步骤. 当时报错的代码大概是这样的:

type TestContent struct {
	Id int
	Content Content // 数据库中的 json 结构
}

type Content struct {
	Name string
	Age int
}

func (c *Content) Scan(value interface{}) error {
	return json.Unmarshal(value.([]byte), c)
}

func (c *Content) Value() (driver.Value, error) {
	return json.Marshal(c)
}

向数据库插入数据, 调用Create方法时报错了:

[2020-08-28 23:18:25] sql: converting argument $1 type: unsupported type main.Content, a struct

这这这, 什么鬼? 当时我百思不得其所. 经过多次尝试, 我发现将Value方法的从属从指针类型改为值类型就可以解决这个问题.

此时我恍然大悟, 想起了之前的方法集的概念.

  1. 指针类型拥有 值/指针 的方法
  2. 值类型只拥有值类型的方法

也就是说, go 在底层是使用值类型来调用的, 所以拿不到指针方法, 故而报错.

看到这里, 如果你也遇到同样的问题, 将Value方法从属改为值类型就可以解决了. 以下内容是我手贱之后的另一个愚蠢记录, 可跳过.

另一个问题

此时我以为我已经深得精髓, 解决方法很简单, 将两个方法的从属都改为值类型就好了嘛. 修改后, 插入数据果然没有问题了, 但是当我查询的时候, 发现了另一个问题, Content对象没有赋值, 是空的.

当时我一脸懵逼, 没有找到问题所在, 我做了什么? 于是, 我就开始了打断点之路:

我发现它走到这里, 调用了Scan方法, 那么, dest 又是个什么对象呢?

于是, 我又找到了这个赋值的地方, 将类型打印出来后, 是:

**main.Content

是一个二级指针, 这时, 我以为是因为二级指针的问题. 于是我动手写了一段代码来模拟这段操作:

func main(){
  // 这里模拟了当时设置的代码内容
	typeOf := reflect.TypeOf(Content{})
	reflectValue := reflect.New(reflect.PtrTo(typeOf))
	reflectValue.Elem().Set(reflect.ValueOf(&Content{}))
	r := reflectValue.Interface()
	if c, ok := r.(**Content); ok {
		(**c).SetName("1111")
		fmt.Println(fmt.Sprintf("%+v", **c))
	}
}

// 这里, 为了方便测试, 添加了 SetName 方法, 与 Scan 相同
func (nt Content) SetName(name string) {
	nt.Name = name
}

当我看到结果的时候, 发现name依旧没有设置进去. 我了个喵, 什么情况?

然后我开始了疯狂检查的过程, 直到我写下了这段代码之后, 我陷入了沉思:

	content := Content{}
	content.SetName("hh")
	fmt.Println(fmt.Sprintf("%+v", content))

当我发现直接设置都没用的时候, 我知道, 一定是我哪个最简单的地方出错了. 我默默的点起一支烟, 望着眼前的代码发起了呆.

我经过与之前改动的对比, 知道问题一定是出在指针与值类型的转换上.

我我我我的天, 最终我发现我犯了一个多么愚蠢的错误. 使用值类型是无法对其字段进行修改的, 其修改通通是通过值复制进行, 并不会影响原始对象. 而且我右打了断点发现, 方法并不是没有调, 确实是调用了, 只不过因为从属与值而没有对原始对象造成影响.

总结

就在我刚开始查这个问题的时候, 我自认为找到了什么不得了的 bug, 满心激动的查了下去. 直到最终发现问题的时候, 我懵逼了.

之前我哥就和我说, 查问题要从表现去推测. 而这次就是直接奔着底层去了, 结果做了很多无用功.

我回想了一下, 当时正确的检查步骤应该是:

  1. Scan方法内打断点, 查看是否调用了方法以及两次调用传的参数是否一致
  2. 当发现调用方法且参数一致时, 就直接到了最后一步并最终找到指针的问题
  3. 若没有调用方法或参数不一致时, 再往调用的地方去找

步骤简单来说, 就是自上而下, 先从外层找问题, 当发现外层一切正常, 再向里边找, 就像剥洋葱一样, 一层一层, 直到定位到问题所在.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: 在 Go 语言中,最受欢迎的 ORM(Object-Relational Mapping)框架可能是 Gorm。Gorm 是一个简单而强大的 ORM 框架,支持许多数据库,包括 MySQL、PostgreSQL、SQLite 和 SQL Server。它具有高度可配置的查询构建器,允许您使用简单的方法查询数据库。此外,Gorm 还提供了许多有用的功能,如自动迁移、钩子函数和缓存支持。 另一个受欢迎的 Go 语言 ORM 框架是 Beego ORM。Beego ORM 是与 Beego 框架一起使用的 ORM,但也可以单独使用。它提供了简单易用的 API,允许您在 Go 中轻松地操作数据库。Beego ORM 还提供了许多有用的功能,如自动迁移、模型钩子函数和查询构建器支持。 还有许多其他优秀的 Go 语言 ORM 框架,如 SQLBoiler、Xorm 和 buf clean。最终,哪个 ORM 框架最适合您取决于您的需求和喜好。建议您比较多个选项,然后根据您的项目需求选择最合适的 ORM 框架。 ### 回答2: 在Go语言中,有许多流行的ORM(对象关系映射)框架可供选择,而最好用的框架往往会因个人需求而异。然而,其中一个备受推崇的是GORM。 GORM是一个功能强大的Go语言ORM框架,被广泛认可的原因有以下几点: 1. 简单易用:GORM提供了简洁直观的API,使得对数据库的操作变得轻而易举。开发者可以利用GORM实现复杂的数据库查询、插入、更新和删除操作,而无需深入学习SQL语句。 2. 数据库兼容性:GORM支持多种数据库,如MySQL、PostgreSQL、SQLite等,因此具备良好的数据库兼容性。这使得GORM可以适用于各种类型的项目,无论是小型还是大型,无论是简单还是复杂。 3. 数据迁移工具:GORM内置了强大的数据库迁移工具,可以轻松地管理数据库模式的变更。通过简单的命令,开发者可以自动执行数据库迁移脚本,而无需手动操作数据库。 4. 提供事务支持:GORM支持事务处理,使得开发者可以对数据库操作进行原子性处理。通过提供Begin、Commit和Rollback等事务相关的方法,GORM确保了数据的完整性和一致性。 5. 强大的查询构建器:GORM的查询构建器使得开发者可以快速构建复杂的查询语句,包括联接查询、条件筛选、分组、排序等。同时,GORM还支持预加载机制,减少了数据库查询的次数,提高了性能。 总而言之,GORM作为一种功能强大、易用性高、数据库兼容性好的ORM框架,可以帮助开发者更加便捷地操作数据库,提高开发效率和代码质量。因此,GORM可以被认为是Go语言中最好用的ORM框架之一。 ### 回答3: Go语言最好用的ORM框架是GORM。 GORM是一个功能丰富且易于使用的ORM库,提供了非常方便的数据库访问和操作接口,可以与多种数据库系统进行交互。以下是GORM的一些特点: 1. 简单易用:GORM提供了简洁的API和强大的查询功能,使得开发人员可以更轻松地进行数据库操作。 2. 自动迁移:GORM可以自动将Go结构体与数据库表进行映射,并可以根据结构体的变化自动更新数据库表结构,减少了手动迁移的工作。 3. 事务支持:GORM支持事务操作,开发人员可以使用事务来保证数据库操作的一致性和完整性。 4. 关联关系映射:GORM支持一对一、一对多、多对多等关联关系的映射,可以轻松进行复杂的数据查询和操作。 5. 高性能:GORM通过一系列的优化措施,如缓存、批量操作等,提供了较高的性能。 6. 扩展性:GORM提供了丰富的Hook机制,可以方便地在数据库操作中插入自定义逻辑,扩展其功能。 总之,GORM是一个功能全面、易于使用且性能优越的ORM框架,它为Go语言的开发者提供了便捷的数据库操作解决方案,是最好用的ORM框架之一。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值