如何用它的值对Map [string] int进行排序?

sam*_*mol 70 go

鉴于此代码块

map[string]int {"hello":10, "foo":20, "bar":20}
Run Code Online (Sandbox Code Playgroud)

我想打印出来

foo, 20
bar, 20
hello, 10
Run Code Online (Sandbox Code Playgroud)

按从高到低的顺序

谢谢!

sam*_*mol 80

在Andrew Gerrand找到了关于Golang-nuts的答案

您可以通过编写len/less/swap函数来实现排序接口

func rankByWordCount(wordFrequencies map[string]int) PairList{
  pl := make(PairList, len(wordFrequencies))
  i := 0
  for k, v := range wordFrequencies {
    pl[i] = Pair{k, v}
    i++
  }
  sort.Sort(sort.Reverse(pl))
  return pl
}

type Pair struct {
  Key string
  Value int
}

type PairList []Pair

func (p PairList) Len() int { return len(p) }
func (p PairList) Less(i, j int) bool { return p[i].Value < p[j].Value }
func (p PairList) Swap(i, j int){ p[i], p[j] = p[j], p[i] }
Run Code Online (Sandbox Code Playgroud)

有关原始帖子,请在此处找到https://groups.google.com/forum/#!topic/golang-nuts/FT7cjmcL7gw

  • @larsmans我的坏!谢谢你指出来.我改为使用sort.Reverse来得到相反的结果 (3认同)
  • 更好的是,我什至不知道 `sort.Reverse`。+1。 (2认同)

vou*_*rus 52

有一个新的sort.Slice函数在1.8,所以现在这更简单.

package main

import (
    "fmt"
    "sort"
)

func main() {
    m := map[string]int{
        "something": 10,
        "yo":        20,
        "blah":      20,
    }

    type kv struct {
        Key   string
        Value int
    }

    var ss []kv
    for k, v := range m {
        ss = append(ss, kv{k, v})
    }

    sort.Slice(ss, func(i, j int) bool {
        return ss[i].Value > ss[j].Value
    })

    for _, kv := range ss {
        fmt.Printf("%s, %d\n", kv.Key, kv.Value)
    }
}
Run Code Online (Sandbox Code Playgroud)

https://play.golang.org/p/y1_WBENH4N

  • 我不喜欢输出不是我开始时的地图 (4认同)
  • 还有一个 [sort.SliceStable](https://golang.org/pkg/sort/#SliceStable) (也在 Go 1.8 中添加)保留了相等元素的原始顺序。 (2认同)

Ale*_*bov 24

首先按值对键进行排序,然后迭代映射:

package main

import (
    "fmt"
    "sort"
)

func main() {
    counts := map[string]int{"hello": 10, "foo": 20, "bar": 20}

    keys := make([]string, 0, len(counts))
    for key := range counts {
        keys = append(keys, key)
    }
    sort.Slice(keys, func(i, j int) bool { return counts[keys[i]] > counts[keys[j]] })

    for _, key := range keys {
        fmt.Printf("%s, %d\n", key, counts[key])
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 我很确定谁在投反对票,可能是那些没有仔细阅读比较器函数的人。还接受的答案不会产生要求的打印输出,而是引入了必须在实际代码中维护的新数据结构。这是我的答案的游乐场链接 https://play.golang.org/p/Y4lrEm2-hT5 (6认同)
  • 这应该是公认的答案,最简单的解决方案。 (2认同)

zzz*_*zzz 16

例如:

package main

import (
        "fmt"
        "sort"
)

func main() {
        m := map[string]int{"hello": 10, "foo": 20, "bar": 20}
        n := map[int][]string{}
        var a []int
        for k, v := range m {
                n[v] = append(n[v], k)
        }
        for k := range n {
                a = append(a, k)
        }
        sort.Sort(sort.Reverse(sort.IntSlice(a)))
        for _, k := range a {
                for _, s := range n[k] {
                        fmt.Printf("%s, %d\n", s, k)
                }
        }
}
Run Code Online (Sandbox Code Playgroud)

操场


输出:

foo, 20
bar, 20
hello, 10
Run Code Online (Sandbox Code Playgroud)

  • @newacct:它只解决 OP 问题,而不是一般情况;-) (2认同)