golang对自定义类型进行排序的解决方法
前言
Go语言支持我们自定义类型,我们大家在实际项目中,常常需要根据一个结构体类型的某个字段进行排序。之前遇到这个问题不知道如何解决,后来在网上搜索了相关问题,找到了一些好的解决方案,此处参考下,做个总结吧。
由于golang的sort包本身就提供了相应的功能,我们就没必要重复的造个轮子了,来看看如何利用sort包来实现吧。
sort包浅谈
golang中也实现了排序算法的包sort包,sort包在内部实现了四种基本的排序算法:插入排序(insertionSort)、归并排序(symMerge)、堆排序(heapSort)和快速排序(quickSort);sort包会依据实际数据自动选择最优的排序算法。
所以我们写代码时只需要考虑实现sort.Interface这个类型就可以了。
粗略的看看sort包
funcSort(dataInterface){
//Switchtoheapsortifdepthof2*ceil(lg(n+1))isreached.
n:=data.Len()
maxDepth:=0
fori:=n;i>0;i>>=1{
maxDepth++
}
maxDepth*=2
quickSort(data,0,n,maxDepth)
}
typeInterfaceinterface{
//Lenisthenumberofelementsinthecollection.
Len()int
//Lessreportswhethertheelementwith
//indexishouldsortbeforetheelementwithindexj.
Less(i,jint)bool
//Swapswapstheelementswithindexesiandj.
Swap(i,jint)
}
//内部实现的四种排序算法
//插入排序
funcinsertionSort(dataInterface,a,bint)
//堆排序
funcheapSort(dataInterface,a,bint)
//快速排序
funcquickSort(dataInterface,a,b,maxDepthint)
//归并排序
funcsymMerge(dataInterface,a,m,bint)
所以要调用sort.Sort()来实现自定义类型排序,只需要我们的类型实现Interface接口类型中的三个方法即可。
先看看sort包本身对于[]int类型如何排序
//首先定义了一个[]int类型的别名IntSlice
typeIntSlice[]int
//获取此slice的长度
func(pIntSlice)Len()int{returnlen(p)}
//比较两个元素大小升序
func(pIntSlice)Less(i,jint)bool{returnp[i]
照葫芦画瓢我们来对自定义的结构体类型进行降序排序
packagemain
import(
"fmt"
"sort"
)
typePersonstruct{
Namestring
Ageint
}
typePersons[]Person
//获取此slice的长度
func(pPersons)Len()int{returnlen(p)}
//根据元素的年龄降序排序(此处按照自己的业务逻辑写)
func(pPersons)Less(i,jint)bool{
returnp[i].Age>p[j].Age
}
//交换数据
func(pPersons)Swap(i,jint){p[i],p[j]=p[j],p[i]}
funcmain(){
persons:=Persons{
{
Name:"test1",
Age:20,
},
{
Name:"test2",
Age:22,
},
{
Name:"test3",
Age:21,
},
}
fmt.Println("排序前")
for_,person:=rangepersons{
fmt.Println(person.Name,":",person.Age)
}
sort.Sort(persons)
fmt.Println("排序后")
for_,person:=rangepersons{
fmt.Println(person.Name,":",person.Age)
}
}
其实,一般Len()和Swap()基本不做改变,只有涉及到元素比较的Less()方法会有所改变。
当我们对某一个结构体中多个字段进行排序时怎么办,难道每排序一个就写下这三个方法么,当然不是。我们可以利用嵌套结构体来解决这个问题。因为嵌套结构体可以继承父结构体的所有属性和方法
比如我想对上面Person的Name字段和Age对要排序,我们可以利用嵌套结构体来改进一下。
packagemain
import(
"fmt"
"sort"
)
typePersonstruct{
Namestring
Ageint
}
typePersons[]Person
//Len()方法和Swap()方法不用变化
//获取此slice的长度
func(pPersons)Len()int{returnlen(p)}
//交换数据
func(pPersons)Swap(i,jint){p[i],p[j]=p[j],p[i]}
//嵌套结构体将继承Person的所有属性和方法
//所以相当于SortByName也实现了Len()和Swap()方法
typeSortByNamestruct{Persons}
//根据元素的姓名长度降序排序(此处按照自己的业务逻辑写)
func(pSortByName)Less(i,jint)bool{
returnlen(p.Persons[i].Name)>len(p.Persons[j].Name)
}
typeSortByAgestruct{Persons}
//根据元素的年龄降序排序(此处按照自己的业务逻辑写)
func(pSortByAge)Less(i,jint)bool{
returnp.Persons[i].Age>p.Persons[j].Age
}
funcmain(){
persons:=Persons{
{
Name:"test123",
Age:20,
},
{
Name:"test1",
Age:22,
},
{
Name:"test12",
Age:21,
},
}
fmt.Println("排序前")
for_,person:=rangepersons{
fmt.Println(person.Name,":",person.Age)
}
sort.Sort(SortByName{persons})
fmt.Println("排序后")
for_,person:=rangepersons{
fmt.Println(person.Name,":",person.Age)
}
}
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对毛票票的支持。