質問するログイン新規登録

回答編集履歴

1

追記

2020/05/07 08:38

投稿

hoshi-takanori
hoshi-takanori

スコア7903

answer CHANGED
@@ -53,4 +53,59 @@
53
53
  5. data にもっと要素があれば、ソートが完了するまで comparator が何度も呼ばれる。
54
54
  6. 最終的に、stableSort はソートされた新しい配列を返す。
55
55
 
56
- という動作になります。(厳密には、stableSort の中で comparator をさらにラップしてますが、説明はパス。)
56
+ という動作になります。
57
+
58
+ ---
59
+
60
+ 追記。ついでなので、stableSort 関数についても解説しておきます。
61
+
62
+ まず、JavaScript の [sort メソッド](https://developer.mozilla.org/ja/docs/Web/JavaScript/Reference/Global_Objects/Array/sort) ですが、二つ問題があります。
63
+ a) 配列を書き換える
64
+ b) ソート順が安定でない
65
+
66
+ a ですが、例えば上記の data に対して data.sort(comparator) を実行すると、配列 data の中身が書き換わります。(ちなみに、data を const で宣言してますが、これはあくまでも data への参照が const なだけで、data の中身は const ではありません。)
67
+ 配列 data の中身が書き変わると元の順序が失われるだけでなく、React (Material UI って確か React でしたよね?) ではこのような破壊的なデータの変更は禁忌とされています。
68
+ 参考: [Reactのべからず集 - Qiita](https://qiita.com/jkr_2255/items/041f238a940f923e4dfc)
69
+
70
+ b ですが、ソート順が安定 (stable) というのは、例えば User の配列を name でソートした時に、name が同じユーザーが複数いれば元の順序を保証する、ということです。残念ながら JavaScript の sort メソッドは stable とは限らない (処理系やブラウザによって異なる) ため、
71
+
72
+ ```TypeScript
73
+ const data2: User[] = [
74
+ { id: 1, name: 'Taro' },
75
+ { id: 2, name: 'Taro' },
76
+ ];
77
+ ```
78
+
79
+ だった場合に data2.sort(comparator) の結果 id: 2 が先になる可能性があります。
80
+ 参考: [安定ソート - Wikipedia](https://ja.wikipedia.org/wiki/%E5%AE%89%E5%AE%9A%E3%82%BD%E3%83%BC%E3%83%88)
81
+
82
+ これらを解決するために定義されたのが stableSort 関数で、(微妙に書き換えました。)
83
+
84
+ ```TypeScript
85
+ function stableSort<T>(array: T[], comparator: (a: T, b: T) => number): T[] {
86
+ const stabilizedThis = array.map((el, index) => [el, index] as [T, number]);
87
+ const stableComparator = (a: [T, number], b: [T, number]): number => {
88
+ const order = comparator(a[0], b[0]);
89
+ if (order !== 0) return order;
90
+ return a[1] - b[1];
91
+ };
92
+ stabilizedThis.sort(stableComparator);
93
+ return stabilizedThis.map((el) => el[0]);
94
+ }
95
+ ```
96
+
97
+ まず stabilizedThis というのは stable 化するために作った新しい配列で、元の array は T の配列ですが、こちらは `[T, number]` というタプルの配列になってます。タプルというのは、意味的には要素名のない構造体のようなもので、最初の要素は T 型の値で、次の要素は number 型の値、ということです。実際には JavaScript の配列 (長さは固定で、要素の位置によって型が異なる!) として実装されています。
98
+ 参考: [[TypeScript] 配列型とタプル型 | HIROs.NET Blog](https://blog.hiros-dot.net/?p=8670)
99
+
100
+ 上記 data2 に対する stabilizedThis の値は、こんな感じになります。
101
+
102
+ ```TypeScript
103
+ const stabilizedThis: [User, number][] = [
104
+ [{ id: 1, name: 'Taro' }, 0],
105
+ [{ id: 2, name: 'Taro' }, 1],
106
+ ];
107
+ ```
108
+
109
+ 次に stableComparator は comparator をラップした関数で、`[T, number]` 型の引数 a, b を比較するものです。最初に comparator を使って a[0] と b[0] つまり元々の配列の要素を比較し、結果が 0 つまり目的のプロパティの値が同じならば、a[1] と b[1] つまり元の配列における要素のインデックスを比較します。これにより、二つの `[T, number]` が同じになることはないため、stable でない sort を使っても結果は stable になります。
110
+
111
+ そして、最終的に stabilizedThis のソート結果から T 型の値だけを取り出した新しい配列を作って返します。これによって元の配列はそのままで新しい配列を返すことになります。