引用类型:指针、slice切片、map、管道chan、interface等都是引用类型,在作为函数参数的时候进行修改会影响到原有的数据。
实现对Hero结构体切片的排序:sort.Sort(data Interface)
Interface里面有三个方法可以实现,其实也就是你想调用系统提供的方法,对结构体切片进行排序,那么就需要实现接口的三个方法。
其实就是在sort函数里面会使用到接口里面提供的排序方法。(其实就是结构体切片类型实现了那个接口的三个方法,那么就可以去调用sort方法来完成对结构体切片的排序)
结构体的切片类型才能存放多个结构体的变量。
软件包: sort func Sort(data Interface)
package main
import (
"fmt"
"math/rand"
"sort"
)
// Hero 声明hero结构体
type Hero struct {
Name string
Age int
}
// HeroSlice 声明hero结构体切片类型
type HeroSlice []Hero
// Len 实现Interface接口,将下面的三个方法都实现了。那么就可以调用Sort包里面的Sort方法了
func (hs HeroSlice) Len() int {
return len(hs)
}
// Less less这个方法就是决定你使用什么标准进行排序
// 按照hero的年龄从小到大排序
func (hs HeroSlice) Less(i, j int) bool {
return hs[i].Age < hs[j].Age
}
func (hs HeroSlice) Swap(i, j int) {
temp := hs[i]
hs[i] = hs[j]
hs[j] = temp
}
func main() {
var heroes HeroSlice
for i := 0; i < 10; i++ {
hero := Hero{
Name: fmt.Sprintf("hero~%d", i),
Age: rand.Intn(100),
}
heroes = append(heroes, hero)
}
for _, v := range heroes {
fmt.Printf("%v", v)
}
//调用sort包里面的sort方法,之所以将切片对应的类型放进去,是因为该类型实现了这三个方法
sort.Sort(heroes)
fmt.Printf("\n排序后的结果..............\n")
for _, v := range heroes {
fmt.Printf("%v", v)
}
}
{hero~0 99}{hero~1 81}{hero~2 46}{hero~3 33}{hero~4 10}{hero~5 72}{hero~6 43}{hero~7 20}{hero~8 43}{hero~9 18}
排序后的结果..............
{hero~4 10}{hero~9 18}{hero~7 20}{hero~3 33}{hero~6 43}{hero~8 43}{hero~2 46}{hero~5 72}{hero~1 81}{hero~0 99}
如果不想对年龄进行排序了,想对姓名进行排序,修改如下:
func (hs HeroSlice) Less(i, j int) bool {
return hs[i].Name < hs[j].Name
}
接口的妙用在于实现了上面的方法,那么其余的事情就交给Sort包里面的Sort方法去做,至于里面如何去调用这三个方法的,就不需要你去管了。
其实就是这种方式帮你开放了接口,你将接口实现了,那么排序就自动点将字段完成。最后修改一下赋值的语句,这样更加简洁。
func (hs HeroSlice) Swap(i, j int) {
hs[i], hs[j] = hs[j], hs[i]
}
上面这种方式就有利于我们实现高质量的代码,开放一些接口让别让去使用,同时将最核心的代码交给最核心的人去写。
将放开的部分开放给别让使用,提供接口。 下面这些方法放开之后就知道按照什么标准排序。能够放开点就放开,不能够放开的就放到内部去。