1. 程式人生 > >[go-linq]-Go的.NET LINQ式查詢方法

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

## 關於我 [我的部落格|文章首發](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)) ## End 本文到此結束,希望對你有幫助