読み込み中...

現場で差がつく!JavaScriptの配列処理、あなたは正しく使い分けていますか?

JavaScriptのforEachとindexを使った初心者向けの徹底解説画像 JS
この記事は約17分で読めます。

【サイト内のコードはご自由に個人利用・商用利用いただけます】

この記事では、プログラムの基礎知識を前提に話を進めています。

説明のためのコードや、サンプルコードもありますので、もちろん初心者でも理解できるように表現してあります。

本記事のサンプルコードを活用して機能追加、目的を達成できるように作ってありますので、是非ご活用ください。

※この記事は、一般的にプロフェッショナルの指標とされる『実務経験10,000時間以上』を満たす現役のプログラマチームによって監修されています。

※Japanシーモアは、常に解説内容のわかりやすさや記事の品質に注力しております。不具合、分かりにくい説明や不適切な表現、動かないコードなど気になることがございましたら、記事の品質向上の為にお問い合わせフォームにてご共有いただけますと幸いです。
(送信された情報は、プライバシーポリシーのもと、厳正に取扱い、処分させていただきます。)

●forEachの基本的な使い方

JavaScriptでループ処理を行う際、forEachメソッドが非常に便利です。

配列の各要素に対して指定したコールバック関数を実行するforEachメソッドは、読みやすく、メンテナンスしやすいコードを書くのに役立ちます。

forEachメソッドの基本的な構文を見てみましょう。

array.forEach(function(element, index, array) {
  // 処理内容
});

この構文では、第1引数に現在処理されている要素の値、第2引数にその要素のインデックス、第3引数にforEachを呼び出した配列自体が渡されます。

第2引数と第3引数はオプションであり、必要に応じて使用できます。

実際にforEachを使用して配列の各要素を出力する簡単な例を見てみましょう。

○サンプルコード1:配列の各要素を出力する

const fruits = ['apple', 'banana', 'orange'];

fruits.forEach(function(fruit) {
  console.log(fruit);
});

この例では、fruits配列の各要素がコンソールに出力されます。

実行結果は次のようになります。

apple
banana
orange

forEachメソッドを使用すると、配列の要素を簡単に1つずつ取り出して処理できます。

また、コールバック関数をアロー関数で書くことで、さらに簡潔に表現できます。

次に、インデックスを使って配列要素にアクセスする例を見てみましょう。

○サンプルコード2:indexを使って配列要素にアクセス

const fruits = ['apple', 'banana', 'orange'];

fruits.forEach(function(fruit, index) {
  console.log(`${index}: ${fruit}`);
});

この例では、forEachの第2引数でインデックスを受け取り、各要素とともに出力しています。

実行結果は次のようになります。

0: apple
1: banana
2: orange

このように、forEachメソッドを使用すると、配列の要素とそのインデックスを簡単に取得できます。

これは、データの処理や表示において非常に有用です。

●forEachとforの違い

JavaScriptにおいてループ処理を行う際、forEachとforは両方とも頻繁に使用される構文です。

それぞれに特徴があり、状況に応じて使い分けることが重要です。

実行速度と可読性の観点から、両者の違いを詳しく見ていきましょう。

一般的に、forループは実行速度が速いと言われています。

forループが単純な構文で条件式と増分式を明示的に書くため、ループの制御がしやすいからです。

一方、forEachはコールバック関数を呼び出すため、関数呼び出しのオーバーヘッドが発生します。

大量の要素を持つ配列に対してループ処理を行った場合、forループのほうが数倍速いというベンチマーク結果も存在します。

しかし、コードの可読性や保守性を考慮すると、forEachを使用したほうが良い場合も多くあります。

forループで配列を処理する例を見てみましょう。

○サンプルコード3:forループで配列を処理

const numbers = [1, 2, 3, 4, 5];

for (let i = 0; i < numbers.length; i++) {
  console.log(numbers[i]);
}

forループでは、カウンター変数iを初期化し、配列の長さ未満である限りループを続けます。

配列の要素にアクセスするには、numbers[i]のようにインデックスを使用する必要があります。

対照的に、forEachを使用すると、配列の要素を直接取得できるため、コードがシンプルで読みやすくなります。

forEachでインデックスを取得する例を見てみましょう。

○サンプルコード4:forEachでのインデックス取得

const numbers = [1, 2, 3, 4, 5];

numbers.forEach((number, index) => {
  console.log(`${index}: ${number}`);
});

forEachのコールバック関数では、要素の値とインデックスを引数で受け取ることができます。

アロー関数を使用すると、さらに簡潔に記述できます。

forEachを使用すると、配列の要素に直接アクセスできるため、コードの意図が明確になります。

ループのカウンター変数や条件式を管理する必要がないので、ループの本質的な処理に集中できるからです。

チームで開発を行う際、forEachを使用したコードのほうが可読性が高く、バグが混入するリスクを減らせる可能性があります。

新しくプロジェクトに参加したメンバーも、処理の流れを追いやすくなるでしょう。

したがって、実行速度よりも可読性や保守性を重視する場合は、forEachを使用することをお勧めします。

ただし、パフォーマンスが極めて重要な場合や、大量のデータを処理する必要がある場合は、forループの使用を検討してみてください。

●連想配列でのforEachの使い方

連想配列、すなわちオブジェクトに対してもforEachを使用することができます。

Object.keys()、Object.values()、Object.entries()といったメソッドと組み合わせることで、キーや値を柔軟に取得できるようになります。

特に、Object.entries()メソッドを使用すると、オブジェクトのキーと値のペアを配列として取得できます。

これは、オブジェクトのプロパティを効率的に処理する際に非常に便利です。

例えば、ユーザー情報を格納したオブジェクトがあるとします。

連想配列のキーと値を取得する例を解説していきましょう。

○サンプルコード5:連想配列のキーと値を取得

const user = {
  name: 'John Smith',
  age: 30,
  email: 'john@example.com'
};

Object.entries(user).forEach(([key, value]) => {
  console.log(`${key}: ${value}`);
});

この例では、Object.entries(user)でキーと値のペアを配列で取得し、forEachのコールバック関数でそれぞれを分割代入しています。

これで、キーと値を個別に扱うことができます。

実行結果は次のようになります。

name: John Smith
age: 30
email: john@example.com

オブジェクトのプロパティを列挙する別の方法として、for…inループを使用することもできます。

for…inループを使用してオブジェクトのプロパティを列挙する例を見てみましょう。

○サンプルコード6:オブジェクトのプロパティを列挙

const user = {
  name: 'John Smith',
  age: 30,
  email: 'john@example.com'
};

for (const key in user) {
  console.log(`${key}: ${user[key]}`);
}

for…inループでは、オブジェクトのキーを順番に取得できます。

ただし、プロトタイプチェーン上のプロパティも列挙されてしまうので、必要に応じてhasOwnProperty()メソッドでフィルタリングする必要があります。

さらに、ネストした連想配列を扱う場合は、再帰的な処理が有効です。

ネストした連想配列を操作する例を見てみましょう。

○ネストした連想配列の操作

const data = {
  users: [
    { id: 1, name: 'John', age: 30 },
    { id: 2, name: 'Alice', age: 25 },
  ],
  posts: [
    { id: 1, title: 'Hello World', authorId: 1 },
    { id: 2, title: 'Introduction', authorId: 2 },
  ],
};

function printNestedObject(obj) {
  Object.entries(obj).forEach(([key, value]) => {
    console.log(`${key}:`);
    if (Array.isArray(value)) {
      value.forEach((item) => {
        printNestedObject(item);
        console.log('---');
      });
    } else if (typeof value === 'object') {
      printNestedObject(value);
    } else {
      console.log(value);
    }
  });
}

printNestedObject(data);

この例では、printNestedObject()関数がオブジェクトのキーと値を再帰的に処理しています。

値が配列の場合はさらにその要素を再帰的に処理し、オブジェクトの場合も同様に処理します。

これで、ネストの深さに関係なく、連想配列のすべての要素にアクセスできます。

このように、連想配列でもforEachを活用することで、柔軟なデータ操作が可能になります。

オブジェクトのキーと値を同時に取得したり、ネストした構造を再帰的に処理したりできるからです。

実際の開発では、外部APIから取得したJSONデータを連想配列として扱うことが多いです。

そのような場合、forEachを使用してデータを加工・整形すれば、アプリケーションで使いやすい形に変換できます。

●アロー関数を使ったforEach

モダンなJavaScriptの書き方の代表格であるアロー関数を使用したforEachの使い方について解説します。

アロー関数を活用することで、よりシンプルで読みやすいコードを書くことができるようになります。

アロー関数は、functionキーワードを使用した従来の関数定義を簡潔に書き換える構文です。

アロー関数を使用すれば、コールバック関数の定義をワンライナーで書くことができるため、コードの見通しが良くなります。

従来の関数定義とアロー関数を比較してみましょう。

○アロー関数の基本構文

従来の関数定義

const double = function(x) {
  return x * 2;
};

アロー関数での定義

const double = (x) => {
  return x * 2;
};

さらに、関数の本体が1つの式だけの場合、ブロックとreturnを省略できます。

const double = (x) => x * 2;

このアロー関数を使用して、forEachのコールバック関数をシンプルに書き換えることができます。

シンプルな記述でループ処理を行う例を見てみましょう。

○サンプルコード7:シンプルな記述でループ処理

const numbers = [1, 2, 3, 4, 5];

numbers.forEach((num) => {
  console.log(num);
});

アロー関数を使用すれば、コールバック関数の定義がワンライナーで済むので、全体的にすっきりとした印象になります。

さらに、アロー関数を使用すれば、特定の条件を満たす要素だけを処理するような場合にも、コードが読みやすくなります。

特定の条件を満たす要素の処理を行う例を見てみましょう。

○サンプルコード8:特定の条件を満たす要素の処理

const numbers = [1, 2, 3, 4, 5];

numbers.forEach((num) => {
  if (num % 2 === 0) {
    console.log(`${num} is even`);
  } else {
    console.log(`${num} is odd`);
  }
});

偶数と奇数で異なる処理を行う場合でも、アロー関数を使用すればif文の中身がすっきりと書けます。

アロー関数を使用したforEachは、コードの可読性を高めつつ、処理の流れを明確にしてくれます。

無駄な記述が減ることで、ループ処理の本質的な部分に集中できるからです。

チームで開発している時、アロー関数を使用したコードなら、他のメンバーも処理の内容を素早く理解できる可能性があります。

コードレビューの時間も短縮できるかもしれません。

慣れないうちは、従来の関数定義とアロー関数を併用しながら、徐々にアロー関数に移行していくことをお勧めします。

●よくあるエラーと対処法

forEachを使用してループ処理を書いた際に、エラーが発生することがあります。代表的なエラーとその対処法について説明します。

「forEach is not a function」というエラーは、forEachを配列ではないオブジェクトに対して使用しようとした時に発生します。

forEachは配列に対して使用するメソッドなので、配列以外のオブジェクトに使用するとエラーになってしまいます。

例えば、nullやundefinedに対してforEachを使用しようとすると、このエラーが発生します。

「forEach is not a function」エラーの原因と解決策を見てみましょう。

○「forEach is not a function」エラーの原因と解決策

const data = null;
data.forEach((item) => {
  console.log(item);
});

このコードでは、nullに対してforEachを使用しようとしているため、エラーが発生します。

解決策としては、事前にデータが配列であることを確認するのが効果的です。

その方法を見てみましょう。

if (Array.isArray(data)) {
  data.forEach((item) => {
    console.log(item);
  });
} else {
  console.log('データが配列ではありません');
}

Array.isArray()メソッドで配列かどうかをチェックすれば、エラーを未然に防ぐことができます。

「Cannot read property ‘forEach’ of undefined」エラーも、似たような原因で発生します。

このエラーは、undefinedに対してforEachを使用しようとした時に起こります。

○「Cannot read property ‘forEach’ of undefined」エラーへの対処

let data;
data.forEach((item) => {
  console.log(item);
});

この場合も、データが存在することを確認してからforEachを使用するのが賢明です。

その方法は次のようになります。

if (data) {
  data.forEach((item) => {
    console.log(item);
  });
} else {
  console.log('データが未定義です');
}

データが未定義の場合は、適切なエラーメッセージを表示するなどの処理を行いましょう。

パフォーマンス面での注意点として、forEachはfor文よりも若干処理速度が遅いとされています。

forEachはコールバック関数を呼び出すため、関数呼び出しのオーバーヘッドが発生するからです。

大量のデータを処理する場合、forEachよりもfor文の方が高速に処理できることがあります。

パフォーマンス面での注意点を見てみましょう。

○パフォーマンス面での注意点

const largeArray = [...Array(100000).keys()];

console.time('forEach');
largeArray.forEach((item) => {
  // 処理内容
});
console.timeEnd('forEach');

console.time('for');
for (let i = 0; i < largeArray.length; i++) {
  // 処理内容
}
console.timeEnd('for');

このようにパフォーマンスを比較すると、forEachよりもfor文の方が若干高速なことがわかります。

ただし、パフォーマンスの差は多くの場合微々たるものなので、可読性を優先してforEachを使用することをお勧めします。

実際の開発では、コードの明確さと保守性が重要となる場合が多いからです。

●forEachの応用例

これまで、forEachの基本的な使い方から、連想配列での活用法、アロー関数との組み合わせまで見てきました。

ここでは、実務でよく使用されるforEachの応用例をいくつか紹介します。

forEachを使用すれば、配列の各要素に対して同じ処理を行うことができます。

例えば、配列の合計値を求めるような場合に活用できます。配列の合計値を求める例を見てみましょう。

○サンプルコード9:配列の合計値を求める

const numbers = [1, 2, 3, 4, 5];
let sum = 0;

numbers.forEach((num) => {
  sum += num;
});

console.log(sum); // 出力結果: 15

forEachを使用して配列の各要素を順番に処理し、合計値を求めています。

変数sumを宣言し、コールバック関数内で各要素を加算していくことで、配列の合計値が得られます。

複数の配列を組み合わせて処理する際にも、forEachが活躍します。

複数の配列を組み合わせて処理する例を見てみましょう。

○サンプルコード10:複数の配列を組み合わせて処理

const names = ['Alice', 'Bob', 'Charlie'];
const ages = [24, 30, 28];

names.forEach((name, index) => {
  console.log(`${name} is ${ages[index]} years old.`);
});

実行結果

Alice is 24 years old.
Bob is 30 years old.
Charlie is 28 years old.

namesとagesという2つの配列を組み合わせて処理しています。

forEachのコールバック関数では、第2引数でインデックスを受け取れるため、それを使用してagesの対応する要素にアクセスしています。

さらに、DOMの要素を操作する際にもforEachが使用できます。

DOMの要素を操作する例をみてみましょう。

○サンプルコード11:DOMの要素を操作

const items = document.querySelectorAll('ul li');

items.forEach((item) => {
  item.addEventListener('click', () => {
    item.classList.toggle('completed');
  });
});

querySelectorAll()で取得したDOMの要素に対して、forEachを使用してクリックイベントを追加しています。

これで、リストの各アイテムをクリックすると、completedクラスが付与または削除されるようになります。

非同期処理との組み合わせも、forEachの得意分野です。

非同期処理との組み合わせの例を見てみましょう。

○サンプルコード12:非同期処理との組み合わせ

const urls = [
  'https://api.example.com/data1',
  'https://api.example.com/data2',
  'https://api.example.com/data3',
];

urls.forEach(async (url) => {
  const response = await fetch(url);
  const data = await response.json();
  console.log(data);
});

URLの配列に対してforEachを使用し、非同期でデータを取得しています。

async/awaitを使用することで、各URLに対するリクエストを順番に処理できます。

このように、forEachは配列の操作だけでなく、様々な場面で活用できる万能なメソッドです。

シンプルな記述で配列の各要素に対して処理を行えるからこそ、多様な応用が可能になります。

実際の開発では、ユーザーの入力データを検証したり、APIから取得したデータを加工したりと、配列の処理が欠かせません。

そのような場面で、forEachを使用すれば、コードの見通しを保ちつつ、効率的に処理を記述できるでしょう。

まずはシンプルな例から始めて、徐々に応用範囲を広げていくのが上達のコツです。

forEachの特性を理解し、適切に活用することで、より洗練されたJavaScriptコードを書くことができるようになります。

まとめ

JavaScriptのforEachとindexの使い分け方について、基本的な使い方から応用例まで幅広く解説してきました。

forEachはシンプルな記述で配列の各要素に対して処理を行えるため、様々な場面で活躍する万能なメソッドです。

従来のforループと比べると実行速度は若干劣りますが、コードの可読性と保守性を重視する場合はforEachを使用することがおすすめです。

アロー関数と組み合わせることで、よりシンプルで読みやすいコードを書くことができます。

連想配列の操作や、DOMの要素の操作、非同期処理との組み合わせなど、forEachの応用範囲は広いので、様々なケースで活用してみてください。

エラー対策にも気を付けながら、JavaScriptを楽しんでいきましょう。