溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

golag如何使用sort.slice包實(shí)現(xiàn)對象list排序

發(fā)布時間:2022-03-25 13:35:48 來源:億速云 閱讀:144 作者:小新 欄目:開發(fā)技術(shù)

這篇文章將為大家詳細(xì)講解有關(guān)golag如何使用sort.slice包實(shí)現(xiàn)對象list排序,小編覺得挺實(shí)用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

1.sort.Sort介紹

使用sort.Slice進(jìn)行排序,因?yàn)閟lice把struct抽象化了,且slice封裝過了,簡單的基礎(chǔ)類型可以使用sort,使用sort排序需要重寫三個interface,不想學(xué)習(xí)sort排序的可以直接看第三步

這里將對比sort跟slice兩種排序的使用方式

1.1分析內(nèi)置sort包

golag如何使用sort.slice包實(shí)現(xiàn)對象list排序

1.2分析sort.go

golag如何使用sort.slice包實(shí)現(xiàn)對象list排序

要想實(shí)現(xiàn)自定義對象的排序功能,需要重寫這三個interface

外部調(diào)用sort方法即可實(shí)現(xiàn)排序

golag如何使用sort.slice包實(shí)現(xiàn)對象list排序

func Sort(data Interface) {
n := data.Len()
quickSort(data, 0, n, maxDepth(n))
}

通過sort源碼可以看到用的快速排序,不懂快排的可以自行重溫快排算法,這里就不講解快排原理了

2.使用方法

2.1基礎(chǔ)類型排序

基礎(chǔ)類型的排序直接在sort.go去找就行了

package main

import (
"fmt"
"sort"
)

func main() {
intList := [] int {2, 4, 3, 5, 7, 6, 9, 8, 1, 0}
float8List := [] float64 {4.2, 5.9, 12.3, 10.0, 50.4, 99.9, 31.4, 27.81828, 3.14}
stringList := [] string {"a", "c", "b", "d", "f", "i", "z", "x", "w", "y"}

sort.Ints(intList)
sort.Float64s(float8List)
sort.Strings(stringList)

fmt.Printf("%v\n%v\n%v\n", intList, float8List, stringList)

}

golag如何使用sort.slice包實(shí)現(xiàn)對象list排序

2.2對象排序(單一字段)

如果對一個struct對象的某一個字段進(jìn)行排序就需要重寫一組len,less,swap接口,如果有多組就需要重寫多組接口。

下面直接看例子:

如果一個對象只有一個字段需要排序就用下面的例子比較簡單

package main

import (
"fmt"
"sort"
)

type Person struct {
Name string
Age int
}

// 按照 Person.Age 從大到小排序
type PersonSlice [] Person

func (a PersonSlice) Len() int { // 重寫 Len() 方法
return len(a)
}
func (a PersonSlice) Swap(i, j int){ // 重寫 Swap() 方法
a[i], a[j] = a[j], a[i]
}
func (a PersonSlice) Less(i, j int) bool { // 重寫 Less() 方法, 從大到小排序
return a[j].Age < a[i].Age
}

func main() {
people := [] Person{
{"zhang san", 12},
{"li si", 30},
{"wang wu", 52},
{"zhao liu", 26},
}

fmt.Println(people)

sort.Sort(PersonSlice(people)) // 按照 Age 的逆序排序
fmt.Println(people)

sort.Sort(sort.Reverse(PersonSlice(people))) // 按照 Age 的升序排序
fmt.Println(people)

}

2.3對象排序(多字段)

多個了SortBy方法可以確定是升序還是降序,同時可以確定排序字段是哪個

package main

import (
"fmt"
"sort"
)

type Person struct {
Name string
Age int
}

type PersonWrapper struct {
people [] Person
by func(p, q * Person) bool
}
//用來判斷升序還是降序
type SortBy func(p, q *Person) bool

func (pw PersonWrapper) Len() int { // 重寫 Len() 方法
return len(pw.people)
}
func (pw PersonWrapper) Swap(i, j int){ // 重寫 Swap() 方法
pw.people[i], pw.people[j] = pw.people[j], pw.people[i]
}
func (pw PersonWrapper) Less(i, j int) bool { // 重寫 Less() 方法
return pw.by(&pw.people[i], &pw.people[j])
}

// 封裝成 SortPerson 方法
func SortPerson(people [] Person, by SortBy){
sort.Sort(PersonWrapper{people, by})
}

func main() {
people := [] Person{
{"zhang san", 12},
{"li si", 30},
{"wang wu", 52},
{"zhao liu", 26},
}

fmt.Println(people)

//推薦用封裝的SortPerson方法
sort.Sort(PersonWrapper{people, func (p, q *Person) bool {
return q.Age < p.Age // Age 遞減排序
}})

fmt.Println(people)

//推薦用這種
SortPerson(people, func (p, q *Person) bool {
return p.Name < q.Name // Name 遞增排序
})

fmt.Println(people)
}

如果不喜歡用sortby方法,也可以重寫多個less、len、swap方法,

下面的方法就調(diào)用起來比較簡單,不用重寫by跟sortby方法,但是上面的比較靈活(理解可能比下面的費(fèi)勁)

package main

import (
"fmt"
"sort"
)

type Person struct {
Name string
Weight int
}

type PersonSlice []Person

func (s PersonSlice) Len() int { return len(s) }
func (s PersonSlice) Swap(i, j int) { s[i], s[j] = s[j], s[i] }

type ByName struct{ PersonSlice } // 將 PersonSlice 包裝起來到 ByName 中

func (s ByName) Less(i, j int) bool { return s.PersonSlice[i].Name < s.PersonSlice[j].Name } // 將 Less 綁定到 ByName 上


type ByWeight struct{ PersonSlice } // 將 PersonSlice 包裝起來到 ByWeight 中
func (s ByWeight) Less(i, j int) bool { return s.PersonSlice[i].Weight < s.PersonSlice[j].Weight } // 將 Less 綁定到 ByWeight 上

func main() {
s := []Person{
{"apple", 12},
{"pear", 20},
{"banana", 50},
{"orange", 87},
{"hello", 34},
{"world", 43},
}

sort.Sort(ByWeight{s})
fmt.Println("People by weight:")
printPeople(s)

sort.Sort(ByName{s})
fmt.Println("\nPeople by name:")
printPeople(s)

}

func printPeople(s []Person) {
for _, o := range s {
fmt.Printf("%-8s (%v)\n", o.Name, o.Weight)
}
}

3.sort.Slice介紹

3.1使用方法

golag如何使用sort.slice包實(shí)現(xiàn)對象list排序

很明顯將對象抽象成interface,而上面對對象的排序,只能是person這種特定對象

golag如何使用sort.slice包實(shí)現(xiàn)對象list排序

調(diào)用的時候有點(diǎn)類似java的重寫compare方法

package main

import (
"fmt"
"sort"
)

type Person struct {
Name string
Weight int
}


func main() {
s := []Person{
{"apple", 12},
{"pear", 20},
{"banana", 50},
{"orange", 87},
{"hello", 34},
{"world", 43},
}

//可以向上面一樣對排序字段封裝,加一個sortby字段傳遞
sort.Slice(s,func(i,j int)bool{
return s[i].Weight < s[j].Weight
})
fmt.Println("People by weight:")
printPeople(s)

sort.Slice(s,func(i,j int)bool{
return s[i].Name < s[j].Name
})
fmt.Println("\nPeople by name:")
printPeople(s)

}

func printPeople(s []Person) {
for _, o := range s {
fmt.Printf("%-8s (%v)\n", o.Name, o.Weight)
}
}

3.2運(yùn)行

golag如何使用sort.slice包實(shí)現(xiàn)對象list排序

關(guān)于“golag如何使用sort.slice包實(shí)現(xiàn)對象list排序”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI