[go-linq]-Go的.NET LINQ式查询方法

happlyfox · · 573 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

## 关于我 [我的博客|文章首发](http://www.zhouhuibo.club) # 开发者的福音,go也支持linq了 ## 坑爹的集合 go在进行集合操作时,有很不舒服的地方,起初我真的是无力吐槽,又苦于找不到一个好的第三方库,只能每次写着重复代码。举个栗子 ``` 类 学生{ 姓名 年龄 性别 } ``` 1、现在有10个学生的数组,如果我要统计所有年龄大于20岁的人,那我需要 一、遍历 二、自定义条件 三、再append数组添加。 2、接着我又要统计性别为男的所有学生,我又要重复上面的步骤。 你说坑爹不坑爹,那是真的坑! ### 吐槽之处 重点是95%的代码都是一样的,只是那百分之几有点区别。对于一名合格的程序员,我是坚决不能忍受这种情况的发生。 ### 解决方法 群里寻求大神帮助 混迹各大论坛,博客寻找有效信息 终于在老夫的不懈努力下,发现了一个库。它就是**go-linq**,使用它,能够解决我对集合大部分的需求,让编程更顺手,让工作更强经劲。 ## 什么是Linq呢? LINQ(发音:Link)是语言级集成查询(Language INtegrated Query) •LINQ是一种用来进行数据访问的编程模型,使得.NET语言可以直接支持数据查询 Linq 是C#编程的一个爽的飞起的语法糖,使用过的人无一不对其高级的特性,偏自然语义化的折服。 ### 那么问题来了 那么问题来了,go有没有类似的东西呢,答案是肯定的,这次是我们马上要说的库**Go-linq**。通过这名称你就知道他是做什么的了。不多说了,直接开Lu。 ## 开始使用Go-Linq ### Go-Linq介绍 一个强大的语言集成查询(LINQ)库的Go。 没有依赖! 使用迭代器模式完成延迟求值 对并发使用是安全的 支持泛型函数,使您的代码更干净,并且没有类型断言 支持数组、片、映射、字符串、通道和自定义集合 ## 使用 ``` go get gopkg.in/ahmetb/go-linq.v3 ``` ``` import . "gopkg.in/ahmetb/go-linq.v3" ``` import . 的意思是直接使用库的方法,而不使用前缀。当然你也可以添加,官方的写法是这样的。 ## 案例 定义一个员工类 ``` type Employee struct { Name string Age int Sex int // 0 男 1 女 WorkYear int //工龄 } ``` 创建不同的列表 ``` func initEmployeeData() []Employee { list := make([]Employee, 0) for i := 0; i < 10; i++ { list = append(list, Employee{ Name: "张" + strconv.Itoa(i%4), Age: 10 + i, Sex: i % 2, WorkYear: 1 + i%3, }) } return list } func initSameEployeeData() []Employee { list := make([]Employee, 0) for i := 0; i < 10; i++ { list = append(list, Employee{ Name: "张一", Age: 10, Sex: i % 2, WorkYear: 1, }) } return list } ``` ### 小试牛刀-distinct去除开始 ``` func distinct() { var manEmpRows []Employee rows := initSameEployeeData() fmt.Println("===性别是男的所有员工列表去重===") From(rows).Distinct().ToSlice(&manEmpRows) fmt.Println(manEmpRows) } ``` ===性别是男的所有员工列表去重=== [{张一 10 0 1} {张一 10 1 1}] 结果非常nice,本来我们需要的繁琐步骤,一个linq就解决了,是不是戝Diao! ## 众里寻他千百度-where过滤 ``` //where 过滤条件 var manEmpRows []Employee fmt.Println("===过滤性别是男的员工===") From(rows).WhereT(func(e Employee) bool { return e.Sex == 0 }).ToSlice(&manEmpRows) fmt.Println(manEmpRows) ``` ### 荣获三甲-take+sort ``` //Take 选取从头开始的几个元素 fmt.Println("===过滤性别是男的员工,只选择前俩个===") From(rows).WhereT(func(e Employee) bool { return e.Sex == 0 }).Take(2).ToSlice(&manEmpRows) fmt.Println(manEmpRows) ``` 排序。单字段排序,多字段组合排序。 ``` //where过滤+排序 fmt.Println("===过滤性别是女的员工,且按照工龄降序排序===") From(rows).WhereT(func(e Employee) bool { return e.Sex == 1 }).OrderByDescendingT(func(e Employee) int { return e.WorkYear }).ToSlice(&manEmpRows) fmt.Printf("%+v\n", manEmpRows) //where 过滤+双重排序 fmt.Println("===过滤性别是女的员工,且按照工龄降序排序,再按照年龄升序排序===") From(rows).WhereT(func(e Employee) bool { return e.Sex == 1 }).OrderByDescendingT(func(e Employee) int { return e.WorkYear }).ThenByT(func(e Employee) int { return e.Age }).ToSlice(&manEmpRows) ``` ### 弱水三千 只取一瓢-Select ``` //只获取元素中的某些字段,list输出 var outputRows []string fmt.Println("===只获取元素中的某些字段,list输出===") From(rows).SelectT(func(e Employee) string { return e.Name }).ToSlice(&outputRows) fmt.Println(outputRows) ``` ### 蜂合蚁聚-聚合 ``` //聚合函数 query := From(rows).SelectT(func(e Employee) int { return e.Age }) fmt.Println(query.Average()) fmt.Println(query.Max()) fmt.Println(query.Min()) fmt.Println(query.Count())[] ``` ### 其他 ``` //获取结构体数组首个元素或者末个 firstItem := From(rows).First() fmt.Println(firstItem) lastItem := From(rows).Last() fmt.Println(lastItem) ``` ### 总结 通过介绍,不知道大家对go-linq有没有了一个简单的认识,对Linq的使用有一个大概的了解。如果有,那就参照编码自己手撸一遍,加强印象。 其他的特性大家自行查看官方说明,还有更多有趣的Linq语法糖等着你探索。 ## 参考资料 [github源码](https://github.com/ahmetb/go-linq) [官方使用说明](https://pkg.go.dev/github.com/ahmetb/go-linq#Comparable) ## 最后 **推荐阅读** [Redis工具收费后新的开源已出现](https://mp.weixin.qq.com/s/-TUp2MKKLD3R0j3xt85NUA) [GitHub上Star最高的工程师技能图谱](https://mp.weixin.qq.com/s/NOmbb0FqUmxQOkCf_YhTDw) [中国程序员最容易发错的单词 ](https://mp.weixin.qq.com/s/ceC0NK62bEz-fWVDdg4j9A) [推荐!!! Markdown图标索引网站](https://mp.weixin.qq.com/s/qYDEC-QBIUzgjxlBxQHrJQ)) 本文到此结束,希望对你有帮助 😃 更多精彩技术文章汇总在我的公众号 **程序员工具集** ,持续更新,欢迎关注订阅收藏。 如果还有什么疑问或者建议,可以多多交流,原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。

有疑问加站长微信联系(非本文作者))

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

573 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传