对于Golang,如何对二维结构体数组进行排序?

huangapple go评论111阅读模式
英文:

Sort 2D array of structs Golang

问题

我想为一个二维结构体切片创建一个一致的排序,我是从一个映射中创建这个二维切片,所以顺序总是不同的。

我的结构体如下:

  1. // Hit 包含了一个命中的数据。
  2. type Hit struct {
  3. Key string `json:"key"`
  4. Data []Field `json:"data"`
  5. }
  6. // Hits 存储了一系列的命中。
  7. type Hits [][]Hit

我想为Hits类型的内容提供一个一致的顺序。

我尝试过:

  1. func (c Hits) Len() int { return len(c) }
  2. func (c Hits) Swap(i, j int) { c[i], c[j] = c[j], c[i] }
  3. func (c Hits) Less(i, j int) bool { return strings.Compare(c[i][0].Key, c[j][0].Key) == -1 }

但结果似乎仍然以随机顺序返回。

我在考虑可能对切片中的每个项进行哈希处理,但想知道是否有更简单的选项。

英文:

I want to create a consistent ordering for a 2D slice of structs, I am creating the 2D slice from a map so the order is always different.

My structs look like

  1. // Hit contains the data for a hit.
  2. type Hit struct {
  3. Key string `json:"key"`
  4. Data []Field `json:"data"`
  5. }
  6. // Hits stores a list of hits.
  7. type Hits [][]Hit

I want to provide a consistent order for the contents of my Hits type.

I have tried:

  1. func (c Hits) Len() int { return len(c) }
  2. func (c Hits) Swap(i, j int) { c[i], c[j] = c[j], c[i] }
  3. func (c Hits) Less(i, j int) bool { return strings.Compare(c[i][0].Key, c[j][0].Key) == -1 }

But the results still seem to come back in random order.

I was thinking of possibly hashing each item in the slice but thought there might be an easier option

答案1

得分: 1

map的迭代顺序是不确定的,因为它是一个哈希表(实际上并不是这样——将具有相同键的项按照相同的顺序插入到两个映射中,它们的迭代顺序将是相同的)。

假设你的映射是map[string]Hit,为了以确定的顺序迭代它,我会枚举映射中的键集,对其进行排序,然后使用排序后的集合来枚举映射。

类似这样的代码:

  1. package main
  2. import (
  3. "fmt"
  4. "sort"
  5. )
  6. type Hit struct {
  7. Key string `json:"key"`
  8. Data []Field `json:"data"`
  9. }
  10. type Field struct {
  11. Value string `json:"value"`
  12. }
  13. func main() {
  14. var mapOfHits = getSomeHits()
  15. var sortedHits = sortHits(mapOfHits)
  16. for _, h := range sortedHits {
  17. fmt.Println(h.Key)
  18. }
  19. }
  20. func getSomeHits() map[string]Hit {
  21. return make(map[string]Hit, 0)
  22. }
  23. func sortHits(m map[string]Hit) []Hit {
  24. keys := make([]string, 0, len(m))
  25. sorted := make([]Hit, 0, len(m))
  26. for k := range m {
  27. keys = append(keys, k)
  28. }
  29. sort.Strings(keys)
  30. for _, k := range keys {
  31. sorted = append(sorted, m[k])
  32. }
  33. return sorted
  34. }

你可以点击这里查看示例代码。

英文:

The order of iteration over a map, because it's a hash table is rather indeterminate (it's not, really — insert items with the same keys in the same exact sequence into 2 maps and the order of iteration for each will be identical).

Assuming that your map is a map[string]Hit, to iterate it over in a determinate order, I would enumerate the set of keys in the map, sort that, and use that sorted set to enumerate the map.

Something like this:

  1. package main
  2. import (
  3. "fmt"
  4. "sort"
  5. )
  6. type Hit struct {
  7. Key string `json:"key"`
  8. Data []Field `json:"data"`
  9. }
  10. type Field struct {
  11. Value string `json:"value"`
  12. }
  13. func main() {
  14. var mapOfHits = getSomeHits()
  15. var sortedHits = sortHits(mapOfHits)
  16. for _, h := range sortedHits {
  17. fmt.Println(h.Key)
  18. }
  19. }
  20. func getSomeHits() map[string]Hit {
  21. return make(map[string]Hit, 0)
  22. }
  23. func sortHits(m map[string]Hit) []Hit {
  24. keys := make([]string, 0, len(m))
  25. sorted := make([]Hit, 0, len(m))
  26. for k := range m {
  27. keys = append(keys, k)
  28. }
  29. sort.Strings(keys)
  30. for _, k := range keys {
  31. sorted = append(sorted, m[k])
  32. }
  33. return sorted
  34. }

huangapple
  • 本文由 发表于 2022年5月13日 01:24:03
  • 转载请务必保留本文链接:https://go.coder-hub.com/72219821.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定