24
19

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

JavaScript: Array.reduce()メソッドで配列要素を入れ替える

Last updated at Posted at 2020-05-22

Array.prototype.reduce()は、応用範囲の広い配列操作のメソッドです。ただ、与えられるパラメータの数も多く、使い方がわかりにくいきらいは否めません。配列要素を入れ替える例がわかりやすそうかと思いついたので採り上げます。

数値要素の合計を求める

簡単な例からはじめます。よく使われるのが、数値の配列要素の合計です。アロー関数式=>で与えたコールバック関数の第1引数(sum)が結果の集計で、第2引数の要素(num)を順に取り出して集計にまとめます。合計の場合には、集計結果に要素の値を加えてゆくということです。

const sum = [0, 1, 2, 3, 4].reduce((sum, num) => sum + num);
console.log(sum);  // 10

要素に処理を加えた新たな配列を返す

Array.prototype.reduce()の第2引数は集計の初期値です。前項の例では省きました。すると、インデックス0の値を初期値として、インデックス1から処理がはじまるのです。でも、集計値が配列などのオブジェクトになる場合には、初期値を与えなければなりません。

数値の配列要素を、それぞれ2乗するのがつぎの例です。集計の配列(resultArray)に要素を加えるときは、Array.prototype.push()を用いるのが通常でしょう。ここではあえて、コールバック関数の第3引数(id)を使ってみました。

const array = [0, 1, 2, 3, 4].reduce((resultArray, num, id) => {
	resultArray[id] = num ** 2;
	return resultArray;
}, []);
console.log(array);  // [0, 1, 4, 9, 16]

もっとも、各配列要素への処理が加えられた同じ要素数の配列を返すなら、Array.prototype.map()メソッドの方が端的です。

const array = [0, 1, 2, 3, 4].map((num) => num ** 2);

条件に合った要素からなる配列を返す

Array.reduce()は、Array.map()と異なり、返す配列の要素数は問われません。たとえば、数値の配列から偶数値だけの配列をつくって返すこともできます。この場合、要素数が変わりますので、集計配列(resultArray)に条件に合った値を加えるのはArray.push()メソッドです。

const array = [0, 1, 2, 3, 4].reduce((resultArray, num) => {
	if (num % 2 === 0) {
		resultArray.push(num);
	}
	return resultArray;
}, []);
console.log(array);  // [0, 2, 4]

ただ、条件に合った値が収められた配列をつくるには、Array.prototype.filter()メソッドが用意されています。

const array = [0, 1, 2, 3, 4].filter((num) => num % 2 === 0);

配列要素を入れ替える

Array.reduce()の処理は一応わかっても、他の簡単なメソッドで済ませられるのならどこがいいのか、と思われるかもしれません。けれど逆に考えれば、Array.map()Array.filter()がなかったとしても、Array.reduce()で同じ処理はできます。それが、応用範囲の広いメソッドだという理由です。

とはいえ、Array.reduce()メソッドのありがたみはわかりません。思いついた例が、配列要素を入れ替える処理です。これまでのメソッドで扱おうとすれば、Array.prototype.splice()を使うことになるでしょう。その場合、要素の削除と挿入のふたつの操作が必要です。いずれも、もとの配列の対象要素からあとのインデックスがずれてしまいます。

この要素入れ替えの処理を、つぎのような関数(replaceArrayElements())として定めてみましょう。

replaceArrayElements(配列, 入れ替え先インデックス, 入れ替えもとインデックス)

Array.reduce()メソッドのコールバック関数は、第4引数にもとの配列が受け取れます。そして、メソッドの戻り値は処理を終えた新たな配列です。つまり、もと配列のインデックスは崩れません。したがって、以下のように入れ替えるインデックスの要素を、差し替えさえすればよいのです。

戻り値の配列(resultArray)には、あえてインデックスによるブラケット[]アクセスもArray.push()メソッドも用いず、スプレッド構文...で新たな配列をつくってみました。これでインデックスのずれを気にすることなく、要素の入れ替えができます。

function replaceArrayElements(array, targetId, sourceId) {
	return array.reduce((resultArray, element, id, originalArray) => [
		...resultArray,
		id === targetId ? originalArray[sourceId] :
		id === sourceId ? originalArray[targetId] :
		element
	], []);
}
const array = replaceArrayElements([0, 1, 2, 3, 4], 3, 1);
console.log(array);  // [0, 3, 2, 1, 4]

Array.reduce()のパラメータすべてを用いて、このメソッドがどう役立つか示せたのではないでしょうか。

もっとも、配列要素の入れ替えというお題については、分割代入を使った方がもう少し簡潔になり、処理負荷も下げられそうです。なお、配列はスプレッド構文...で複製しました。

function replaceArrayElements(array, targetId, sourceId) {
	const cloneArray = [...array];
	[cloneArray[targetId], cloneArray[sourceId]] = [array[sourceId], array[targetId]];
	return cloneArray;
}
24
19
2

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
24
19

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?