0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Map の辞書順ソートでハマらない方法

Last updated at Posted at 2025-03-14

Paizaの問題を解いていたら、「攻撃された人のダメージを記録し、名前の辞書順で出力せよ!」という指示に見事にハマった。Map は順番を保持しないので、そのまま forEach() すると順番が崩壊する!辞書順って何!? Map にソート機能がない!?試行錯誤の末、「キーを配列化してソートすればいい」 という学びを得たので、その知見をシェアする!


問題概要
攻撃された人のダメージを記録し、名前の辞書順で出力。

入力例:

3     //n
PAIIZA
PAIZA
PAIIIZA
2     //m
PAIIZA 2
PAIIIZA 3

出力例:

3
2
0




NG例: Map に頼りすぎたコード

const damageMap = new Map();
for (let i = 1; i <= n; i++) {
    damageMap.set(lines[i], 0);
}
for (let i = n + 2; i < n + 2 + m; i++) {
    let [person, damage] = lines[i].split(" ");
    damageMap.set(person, damageMap.get(person) + Number(damage));
}
damageMap.forEach((value, key) => console.log(value)); // 辞書順にならない!

ミスのポイント
✅ Map の挿入順に保証されている(辞書順にならない)



OK例1: Array.from() でソート

const sortedNames = Array.from(damageMap.keys()).sort();
sortedNames.forEach(name => console.log(damageMap.get(name)));

✅ Array.from() でキーを取り出し、sort() で辞書順ソート
✅ get() を使い、順番に出力



OK例2: slice() で無駄を省く

const sortedNames = lines.slice(1, n + 1).sort();
sortedNames.forEach(name => console.log(damageMap.get(name)));

✅ すでに lines に名前があるので Map.keys() 取得不要
✅ キーアクセス回数が減り、わずかにパフォーマンス向上



まとめ
Map は順序を保証しない!辞書順ソートが必要なら、

・Array.from(damageMap.keys()).sort()
・もともとのリストを slice().sort() で利用

これで Map の罠にはまらない!

僕の失敗談と解決話!

0
0
3

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?