结构体切片排序实现

借助标准库Sort方法

func Sort(data Interface)
Sort排序data。它调用1次data.Len确定长度,调用O(n*log(n))次data.Less和data.Swap。
本函数不能保证排序的稳定性(即不保证相等元素的相对次序不变)。

  1. type Interface interface {
  2. // Len方法返回集合中的元素个数
  3. Len() int
  4. // Less方法报告索引i的元素是否比索引j的元素小
  5. Less(i, j int) bool
  6. // Swap方法交换索引i和j的两个元素
  7. Swap(i, j int)
  8. }

代码

函数式编程 - 语法糖
函数封装,交给底层进行算法调用处理
用户自定义行为,灵活,但也带来一些别的问题。

  1. // 接口强大功能初体验
  2. // 函数式编程,只需要实现方法,内部会根据方法的实现处理
  3. package main
  4. import (
  5. "fmt"
  6. "math/rand"
  7. "sort"
  8. )
  9. // 1、声明 Hero 结构体
  10. type Hero struct {
  11. Name string
  12. Age int
  13. }
  14. // 2、声明 Hero 结构体切片,动态增长
  15. type HeroSlice []Hero
  16. // 3、实现 Sort() 接口接收者需要的三个方法 Len Less Swap
  17. func (hs HeroSlice) Len() int {
  18. return len(hs)
  19. }
  20. // Less() 方法决定使用什么标准排序
  21. // 按 hero 的年龄从小到大排序
  22. func (hs HeroSlice) Less(i, j int) bool {
  23. return hs[i].Age < hs[j].Age
  24. // 按照 Name 排序
  25. // return hs[i].Name < hs[j].Name
  26. }
  27. // 交换顺序
  28. func (hs HeroSlice) Swap(i, j int) {
  29. /* temp := hs[i]
  30. hs[i] = hs[j]
  31. hs[j] = temp */
  32. // 以下是Go语法交换顺序
  33. hs[i], hs[j] = hs[j], hs[i]
  34. }
  35. func main() {
  36. // 切片排序
  37. // 先定义一个数组/切片
  38. var intSlice = []int{0, -1, 10, 7, 90}
  39. fmt.Println("intSlice = ", intSlice) // [0 -1 10 7 90]
  40. // 1、冒泡排序
  41. // 2、使用系统提供的方法
  42. // func Ints(a []int)
  43. sort.Ints(intSlice) // Ints函数将a排序为递增顺序。
  44. fmt.Println("sort intSlice = ", intSlice) // [-1 0 7 10 90]
  45. // 结构体切片排序
  46. // 1、冒泡排序
  47. // 2、使用系统提供的方法
  48. /*
  49. sort包
  50. func Sort(data Interface)
  51. Sort排序data。它调用1次 data.Len 确定长度,调用O(n*log(n))次 data.Less 和 data.Swap
  52. 本函数不能保证排序的稳定性(即不保证相等元素的相对次序不变)。
  53. */
  54. // 定义 HeroSlice
  55. var heros HeroSlice
  56. for i := 0; i < 10; i++ {
  57. // 1、生成 Hero 结构体实例
  58. hero := Hero{
  59. Name: fmt.Sprintf("英雄~%d", rand.Intn(100)),
  60. Age: rand.Intn(100),
  61. }
  62. // 2、将生成的 hero 实例放入 heros切片实例
  63. heros = append(heros, hero)
  64. }
  65. // 排序前
  66. for _, v := range heros {
  67. fmt.Println(v)
  68. }
  69. fmt.Println("---------Sort() 方法排序后---------")
  70. // HeroSlice 结构体已全部实现接口的三个方法, 调用 Sort() 排序
  71. sort.Sort(heros) // 切片是引用类型,Sort() 方法内部修改 heros 会作用到源数据
  72. for _, v := range heros {
  73. fmt.Println(v)
  74. }
  75. // 交换值
  76. i := 10
  77. j := 20
  78. i, j = j, i
  79. fmt.Println("i = ", i, "j = ", j) // i = 20 j = 10
  80. }