当前位置:首页 > 科技  > 软件

Find、Take、First和Last函数的区别

来源: 责编: 时间:2023-10-19 09:28:06 184观看
导读大家好,我是渔夫子。在gorm中,要想从数据库中查找数据有多种方法,可以通过Find、Take和First来查找。但它们之间又有一些不同。本文就详细介绍下他们之间的不同。一、准备工作首先我们有一个m_tests表,其中id字段是自增的

大家好,我是渔夫子。bO228资讯网——每日最新资讯28at.com

在gorm中,要想从数据库中查找数据有多种方法,可以通过Find、Take和First来查找。但它们之间又有一些不同。本文就详细介绍下他们之间的不同。bO228资讯网——每日最新资讯28at.com

一、准备工作

首先我们有一个m_tests表,其中id字段是自增的主键,同时该表里有3条数据。如下:bO228资讯网——每日最新资讯28at.com

CREATE TABLE `m_tests` (  `id` bigint(20) NOT NULL AUTO_INCREMENT,  `name` varchar(100) NOT NULL DEFAULT '' COMMENT '姓名',  PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8mb4;INSERT INTO test01.m_test (id,name) VALUES (1,'John'), (2,'Jack'),(3,'David');

基于这个表,我们来看看这几个函数查询出来的结果是什么。bO228资讯网——每日最新资讯28at.com

二、First函数

我们通过ToSql函数将First函数转成对应的sql语句来看。如下:bO228资讯网——每日最新资讯28at.com

func main() {	dsn := "username:password@tcp(127.0.0.1:3306)/test01?charset=utf8mb4&parseTime=True&loc=Local&timeout=1000ms"	config := &gorm.Config{		NamingStrategy: schema.NamingStrategy{			SingularTable: true, // 禁用表名复数		}}	db, _ := gorm.Open(mysql.Open(dsn), config)	var row MTest	sql := db.ToSQL(func(tx *gorm.DB) *gorm.DB {		return tx.First(&row)	})	fmt.Printf("接收的sql语句:%s/n", sql)}

通过该程序,可以看到最终的sql语句如下:bO228资讯网——每日最新资讯28at.com

接收的sql语句:SELECT * FROM `m_test` ORDER BY `m_test`.`id` LIMIT 1

发现First函数是通过主键排序后,只获取一条数据。我们在通过explain来解释一下该条语句:bO228资讯网——每日最新资讯28at.com

explain SELECT * FROM `m_test` ORDER BY `m_test`.`id` LIMIT 1

其输出结果如下:bO228资讯网——每日最新资讯28at.com

也就是说在查询的时候也只扫描一行数据。也就是说First函数只扫描一行数据。bO228资讯网——每日最新资讯28at.com

三、Last函数

同样,我们还是通过ToSQL来讲Last函数转化的sql语句打印出来:bO228资讯网——每日最新资讯28at.com

func main() {	dsn := "username:password@tcp(127.0.0.1:3306)/test01?charset=utf8mb4&parseTime=True&loc=Local&timeout=1000ms"	config := &gorm.Config{		NamingStrategy: schema.NamingStrategy{			SingularTable: true, // 禁用表名复数		}}	db, _ := gorm.Open(mysql.Open(dsn), config)	var rows []MTest	sql := db.ToSQL(func(tx *gorm.DB) *gorm.DB {		return tx.Last(&rows)	})	fmt.Printf("接收的sql语句:%s/n", sql)  db.Last(&rows)  fmt.Printf("最终接收:%+v/n", rows)}

我们看到Last转换成的sql语句如下:bO228资讯网——每日最新资讯28at.com

接收的sql语句:SELECT * FROM `m_test` ORDER BY `m_test`.`id` DESC LIMIT 1

所以,Take实际上是按主键倒序排列,并且只获取1行数据的一个sql。bO228资讯网——每日最新资讯28at.com

我们再看最终获取的结果rows,虽然是个数组,但也只有一行数据。:bO228资讯网——每日最新资讯28at.com

最终结果数据:[{Id:6 Name:}]

所以,Last和First的相同点在于只扫描到表的一条目标数据后就截止了,并赋值给接收变量。不同点在于First是按主键正序排列,Last是按主键倒序排列。bO228资讯网——每日最新资讯28at.com

四、Take函数

再来看看Take函数的执行过程。如下:bO228资讯网——每日最新资讯28at.com

func main() {	dsn := "username:password@tcp(127.0.0.1:3306)/test01?charset=utf8mb4&parseTime=True&loc=Local&timeout=1000ms"	config := &gorm.Config{		NamingStrategy: schema.NamingStrategy{			SingularTable: true, // 禁用表名复数		}}	db, _ := gorm.Open(mysql.Open(dsn), config)	var row MTest	sql := db.ToSQL(func(tx *gorm.DB) *gorm.DB {		return tx.Take(&row)	})	fmt.Printf("接收的sql语句:%s/n", sql)}

Take函数执行时最终转换成的sql语句如下:bO228资讯网——每日最新资讯28at.com

SELECT * FROM `m_test` LIMIT 1

也是只获取一行数据,但和First不同的是缺少了Order BY m_test.id``。bO228资讯网——每日最新资讯28at.com

我们再通过explain来解释下该条语句,如下, type列是ALL,rows列是3,因为我们表里只有3行数据。是全表扫描,然后再随机获取一行数据。如下:bO228资讯网——每日最新资讯28at.com

mysql> explain SELECT * FROM `m_test` LIMIT 1;+----+-------------+--------+------------+------+---------------+------+---------+------+------+----------+-------+| id | select_type | table  | partitions | type | possible_keys | key  | key_len | ref  | rows | filtered | Extra |+----+-------------+--------+------------+------+---------------+------+---------+------+------+----------+-------+|  1 | SIMPLE      | m_test | NULL       | ALL  | NULL          | NULL | NULL    | NULL |    3 |   100.00 | NULL  |+----+-------------+--------+------------+------+---------------+------+---------+------+------+----------+-------+1 row in set, 1 warning (0.09 sec)

所以,Take函数是扫描全表,并随机获取一条数据。所以,Take函数要比First函数性能差。bO228资讯网——每日最新资讯28at.com

同时,我们注意到,因为在sql语句中可以看到都有LIMIT 1的限制,所以Take和First都只能获取一条数据,即便是给传递了一个数组,也只能获取一行数据,不能获取多行数据。bO228资讯网——每日最新资讯28at.com

五、Find函数

再来看看Take函数的执行过程。我们首先给Find函数传递一个普通的非切片变量,如下:bO228资讯网——每日最新资讯28at.com

func main() {	dsn := "username:password@tcp(127.0.0.1:3306)/test01?charset=utf8mb4&parseTime=True&loc=Local&timeout=1000ms"	config := &gorm.Config{		NamingStrategy: schema.NamingStrategy{			SingularTable: true, // 禁用表名复数		}}	db, _ := gorm.Open(mysql.Open(dsn), config)	var row MTest	sql := db.ToSQL(func(tx *gorm.DB) *gorm.DB {		return tx.Find(&row)	})	fmt.Printf("接收的sql语句:%s/n", sql)}

转换成的sql语句如下:bO228资讯网——每日最新资讯28at.com

接收的sql语句:SELECT * FROM `m_test`

和First和Take相比,缺少了Order子句和Limit子句。扫描的是整个表,获取的也是表的所有数据,但因为接收者是一个非切片变量,所以最终只接收了一行数据到row中。bO228资讯网——每日最新资讯28at.com

我们再来看看给Find传递一个切片变量来接收的情况:bO228资讯网——每日最新资讯28at.com

func main() {	dsn := "username:password@tcp(127.0.0.1:3306)/test01?charset=utf8mb4&parseTime=True&loc=Local&timeout=1000ms"	config := &gorm.Config{		NamingStrategy: schema.NamingStrategy{			SingularTable: true, // 禁用表名复数		}}	db, _ := gorm.Open(mysql.Open(dsn), config)	var rows []MTest	tx.Find(&rows)	fmt.Printf("rows:%+v/n", rows)}

这个结果是接收所有查找到的行的数据到rows中。所以大家一定要注意,在使用Find查询的时候一定要加Where条件和查询的数量,以避免扫描和查询全表的数据,尤其是在大数量的表中。bO228资讯网——每日最新资讯28at.com

六、总结

本文主要讲解了First、Last、Take和Find查询函数的不同之处。希望在使用过程中大家根据自己的应用场景选择合适的函数。bO228资讯网——每日最新资讯28at.com

本文链接://www.dmpip.com//www.dmpip.com/showinfo-26-14122-0.htmlFind、Take、First和Last函数的区别

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 压测工具界的 “悍马” :wrk 使用

下一篇: Java中的代码重构:技巧、优秀实践与方法

标签:
  • 热门焦点
  • K60 Pro官方停产 第三方瞬间涨价

    K60 Pro官方停产 第三方瞬间涨价

    虽然没有官方宣布,但Redmi的一些高管也已经透露了,Redmi K60 Pro已经停产且不会补货,这一切都是为了即将到来的K60 Ultra铺路,属于厂家的正常操作。但有意思的是该机在停产之后
  • MIX Fold3包装盒泄露 新机本月登场

    MIX Fold3包装盒泄露 新机本月登场

    小米的全新折叠屏旗舰MIX Fold3将于本月发布,近日该机的真机包装盒在网上泄露。从图上来看,新的MIX Fold3包装盒在外观设计方面延续了之前的方案,变化不大,这也是目前小米旗舰
  • 掘力计划第 20 期:Flutter 混合开发的混乱之治

    掘力计划第 20 期:Flutter 混合开发的混乱之治

    在掘力计划系列活动第20场,《Flutter 开发实战详解》作者,掘金优秀作者,Github GSY 系列目负责人恋猫的小郭分享了Flutter 混合开发的混乱之治。Flutter 基于自研的 Skia 引擎
  • 企业采用CRM系统的11个好处

    企业采用CRM系统的11个好处

    客户关系管理(CRM)软件可以为企业提供很多的好处,从客户保留到提高生产力。  CRM软件用于企业收集客户互动,以改善客户体验和满意度。  CRM软件市场规模如今超过580
  • 十个简单但很有用的Python装饰器

    十个简单但很有用的Python装饰器

    装饰器(Decorators)是Python中一种强大而灵活的功能,用于修改或增强函数或类的行为。装饰器本质上是一个函数,它接受另一个函数或类作为参数,并返回一个新的函数或类。它们通常用
  • 猿辅导与新东方的两种“归途”

    猿辅导与新东方的两种“归途”

    作者|卓心月 出品|零态LT(ID:LingTai_LT)如何成为一家伟大企业?答案一定是对“势”的把握,这其中最关键的当属对企业战略的制定,且能够站在未来看现在,即使这其中的
  • 苹果、三星、惠普等暂停向印度出口笔记本和平板电脑

    苹果、三星、惠普等暂停向印度出口笔记本和平板电脑

    集微网消息,据彭博社报道,在8月3日印度突然禁止在没有许可证的情况下向印度进口电脑/平板及显示器等产品后,苹果、三星电子和惠普等大公司暂停向印度
  • OPPO K11评测:旗舰级IMX890加持 2000元档最强影像手机

    OPPO K11评测:旗舰级IMX890加持 2000元档最强影像手机

    【Techweb评测】中端机型用户群体巨大,占了中国目前手机市场的大头,一直以来都是各手机品牌的“必争之地”,其中OPPO K系列机型一直以来都以高品质、
  • 荣耀Magic4 至臻版 首创智慧隐私通话 强劲影音系统

    荣耀Magic4 至臻版 首创智慧隐私通话 强劲影音系统

    2022年第一季度临近尾声,在该季度内,许多品牌陆续发布自己的最新产品,让大家从全新的角度来了解当今的手机技术。手机是电子设备中,更新迭代十分迅速的一款产品,基
Top
Baidu
map