Map [string] intをその値でソートする方法は?


81

このコードブロックを考えると

map[string]int {"hello":10, "foo":20, "bar":20}

印刷したい

foo, 20
bar, 20
hello, 10

高いものから低いものの順に

ありがとう!

回答:


92

AndrewGerrandによる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] }

元の投稿については、こちらをご覧くださいhttps://groups.google.com/forum/#!topic/golang-nuts/FT7cjmcL7gw


1
...Less間違った結果を返すことを除いて。逆ソートの場合は、を使用します>
フレッドフー

3
@larsmans私の悪い!ご指摘いただきありがとうございます。代わりに、sort.Reverseを使用して逆の結果を取得しました
samol 2013

2
さらに良いことに、私はについてさえ知りませんでしたsort.Reverse。+1。
フレッドフー

72

go 1.8には新しいsort.Slice関数があるので、これはより簡単になりました。

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)
    }
}

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


私は、出力は私が始まったマップではありませんどのように好きではない
ヘンドリー

@hendryこの回答は、特に元の質問の形式に対応しています。go1.12であなただけの地図を印刷することができ、それがソートされます、問題を参照してください。github.com/golang/go/issues/21095
voutasaurus

ほぼ完璧です。同じ値の要素を処理することをお勧めします。
TommasoBarbugli20年

@TommasoBarbugliどうやって?それらを安定させますか?またはアルファベット順に並べ替えますか?Goはマップの反復順序をランダム化して、コンパイラにとって重要ではないと明示的に呼び出される順序に依存しないようにするため、安定性は不可能です。あなたは簡単にsort.Sliceに渡された匿名関数変更することができ、アルファベットの場合
voutasaurus

1
等しい要素の元の順序を保持するsort.SliceStable(Go 1.8でも追加)もあります。
DaveYarwood20年

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)
                }
        }
}

遊び場


出力:

foo, 20
bar, 20
hello, 10

@DarshanComputing:ありがとう、修正されました。
zzzz 2013

1
これは、値にIDがないことを前提としています。
newacct 2013

2
@newacct:一般的なケースではなく、OPの問題のみを解決します;-)
zzzz 2013

このソリューションは、私の場合にも機能し、理解しやすいものです。
トミー

1

私はしばしばmap[string]int私が数えているもののソートをする必要があり、以下を使用しています。

func rankMapStringInt(values map[string]int) []string {
    type kv struct {
        Key   string
        Value int
    }
    var ss []kv
    for k, v := range values {
        ss = append(ss, kv{k, v})
    }
    sort.Slice(ss, func(i, j int) bool {
        return ss[i].Value > ss[j].Value
    })
    ranked := make([]string, len(values))
    for i, kv := range ss {
        ranked[i] = kv.Key
    }
    return ranked
}

これを使用して、値の順にキーを繰り返し処理します

values := map[string]int{"foo": 10, "bar": 20, "baz": 1}

for i, index := range rankMapStringInt(values) {
    fmt.Printf("%3d: %s -> %d", i, index, values[index])
}

0

私の場合、自分で作成したプログラムを扱っていました。このプログラムでは、私は、ちょうどあなたのような地図を作成したとstringしてint。それから私はあなたのように、Goにはこのようなものをソートする組み込みの方法が実際にはないことを発見しました。私は他の答えを読みましたが、私が読んだものが本当に好きではありませんでした。

それで、私は問題について別の方法で考えようとしました。Goは 、スライスでsort.Intsを使用できます。また、Goはカスタムコンパレータでsort.Sliceを使用できます。だからではなく、地図を作成するstringint、私が作成したstructstringint。次に、並べ替えることができます。

package main

import (
   "fmt"
   "sort"
)

type File struct {
   Name string
   Size int
}

func main() {
   a := []File{{"april.txt", 9}, {"may.txt", 7}}
   f := func (n, n1 int) bool {
      return a[n].Size < a[n1].Size
   }
   sort.Slice(a, f)
   fmt.Println(a)
}

他の誰かが作成したマップを処理することを余儀なくされる可能性があるため、これはすべての人に役立つわけではありません。しかし、それは私にとって役に立ちました。良い部分は、他のすべての答えとは異なり、これはループを使用しないことです。


-1

最初に値でキーを並べ替えてから、マップを繰り返します。

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])
    }
}

1
誰が反対票を投じているのか、おそらくコンパレーター機能を注意深く読んでいない人たちだと私は確信しています。また、受け入れられた回答は、要求された出力OPを生成しませんが、代わりに実際のコードで維持する必要がある新しいデータ構造を導入します。ここに私の答えのための遊び場リンクですplay.golang.org/p/Y4lrEm2-hT5
AlexanderYastrebov
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.