カテゴリ: Java 更新日: 2025/12/17
PR
独学でJavaを学んでいる方向け
「実務レベルに到達できるか不安」「1人だと詰まることが多い」場合は、 実践重視で学べる環境を一度確認しておくのも一つの手です。
EBAエデュケーション |学習内容・サポートを見る

Javaのラムダ式forEachでaddはNG?副作用と安全なcollectの基本を解説

【ラムダ式】forEachでaddはアリ?副作用と安全なcollect設計
【ラムダ式】forEachでaddはアリ?副作用と安全なcollect設計

先生と生徒の会話形式で理解しよう

生徒

「Javaのラムダ式でforEachを使って、別のリストにaddするのってダメなんですか?」

先生

「うん、それはよくある疑問だね。forEachaddするのは一見便利そうだけど、実は副作用があるから注意が必要なんだよ。」

生徒

「えっ?でも処理は動くように見えますけど?」

先生

「じゃあ実際のコードを見ながら、ラムダ式の副作用と安全なcollectの使い方を学んでみよう。」

1. forEachでaddするコードの例

1. forEachでaddするコードの例
1. forEachでaddするコードの例

Javaのラムダ式を使って、あるリストから条件に合う要素だけを別のリストに追加するというケースはよくあります。例えば次のようなコードです。


import java.util.*;
import java.util.stream.*;

public class ForEachAddExample {
    public static void main(String[] args) {
        List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);
        List<Integer> evens = new ArrayList<>();
        numbers.stream()
               .filter(n -> n % 2 == 0)
               .forEach(n -> evens.add(n));
        System.out.println(evens);
    }
}

[2, 4]

このコードは一見正しく動いているように見えます。しかし、ラムダ式の中でaddするのは副作用(外部状態の変更)を伴っており、可読性やメンテナンス性の観点から推奨されません。

2. ラムダ式における副作用とは?

2. ラムダ式における副作用とは?
2. ラムダ式における副作用とは?

副作用とは、メソッドやラムダ式が外部の状態を変更してしまうことを指します。関数型プログラミングでは副作用のない純粋な処理が好まれます。理由は以下の通りです。

  • 処理の予測がしやすくなる
  • スレッドセーフになる(並列処理でも安全)
  • バグが起きにくくなる

JavaのStream APIでは、なるべく副作用を避け、データを変換・収集するにはcollectを使うのが基本です。

PR

将来を見据えて、+αのスキルを身につけたい方へ

JavaやLinuxを学んでいても、「このままで市場価値は上がるのか」 「キャリアの選択肢を広げたい」と感じる方は少なくありません。

AIを学ぶならアイデミープレミアム

3. 安全なcollectによるリストの生成

3. 安全なcollectによるリストの生成
3. 安全なcollectによるリストの生成

同じように偶数だけを新しいリストに追加したい場合、以下のようにcollectメソッドを使うと安全かつスッキリ書けます。


List<Integer> evens = numbers.stream()
                              .filter(n -> n % 2 == 0)
                              .collect(Collectors.toList());

Collectors.toList()は、ストリームの要素をまとめて新しいリストに変換してくれます。この方法なら副作用がなく、コードの意味も明確です。

4. forEachとcollectの違いを理解しよう

4. forEachとcollectの違いを理解しよう
4. forEachとcollectの違いを理解しよう

ここで、forEachcollectの違いを簡単に表にまとめます。

項目 forEach collect
用途 要素に処理を適用 要素を集める
副作用 あり(addなど) なし(原則)
コードの明確さ やや低い 高い
並列処理の安全性 注意が必要 安全

5. Javaのラムダ式では副作用を避けるのが基本

5. Javaのラムダ式では副作用を避けるのが基本
5. Javaのラムダ式では副作用を避けるのが基本

Javaのラムダ式を使うと、コードがコンパクトで読みやすくなりますが、その一方で副作用を起こしやすくなる場面もあります。特にforEachの中で外部のリストにaddするような処理は、状態の変化がわかりにくくなるため注意が必要です。

副作用を避けるためには、collectを使って結果を作るように意識すると、安全で再利用性の高いコードになります。

6. 自作のCollectorでより柔軟に収集

6. 自作のCollectorでより柔軟に収集
6. 自作のCollectorでより柔軟に収集

もし標準のCollectors.toList()では対応しきれない特殊な処理が必要な場合、自分でCollectorを作ることもできます。


List<String> upperNames = names.stream()
                                .map(String::toUpperCase)
                                .collect(
                                    ArrayList::new,
                                    ArrayList::add,
                                    ArrayList::addAll
                                );

このように、ラムダ式の使い方に慣れてきたら、collectを柔軟に活用してカスタマイズすることも可能です。

7. forEachで副作用を避けられない場面もある

7. forEachで副作用を避けられない場面もある
7. forEachで副作用を避けられない場面もある

ただし、すべてのforEachがダメというわけではありません。例えばデバッグ用のログ出力や、UI部品への表示処理など、純粋な「出力」に関わる場面では副作用を避けることはできません。

その場合でも、処理の目的が明確であれば問題ありません。ただし、データの加工や変換のような場面では、副作用のあるforEachよりmapcollectを使う方が適切です。

8. Javaラムダ式の正しい使い方を身につけよう

8. Javaラムダ式の正しい使い方を身につけよう
8. Javaラムダ式の正しい使い方を身につけよう

Javaのラムダ式はとても便利で、forEachやmap、filterなどを使って柔軟なデータ操作ができます。しかし、副作用を伴う書き方をしてしまうと、後でエラーが発生したときに原因がわかりにくくなります。

副作用を避け、データの変換や収集にはcollectを使うことを基本とすることで、可読性が高く、安全なコードを実現できます。

JavaのStream APIとラムダ式を正しく使いこなして、安全でバグの少ないコーディングを目指しましょう。

まとめ

まとめ
まとめ

Javaのラムダ式forEachとaddの関係を振り返る

この記事では、Javaのラムダ式とStream APIにおけるforEachの使い方を中心に、なぜforEachの中でaddする書き方が推奨されないのか、その理由と背景を丁寧に解説してきました。Javaでリスト操作やデータ加工を行う際、直感的に書けてしまうコードほど、実は注意が必要なケースが多く存在します。

特にforEachを使って別のリストにaddする書き方は、一見すると問題なく動作し、初心者の方ほど「これで良いのでは」と感じやすいポイントです。しかし、その内部では外部の状態を変更する副作用が発生しており、コードの意図が分かりにくくなったり、将来的な修正や拡張の際にバグを生みやすくなったりします。

副作用とは何かを正しく理解する

ラムダ式における副作用とは、処理の結果として外部の変数やコレクションの状態を変更してしまうことを指します。JavaのStream APIは、もともと関数型プログラミングの考え方を取り入れて設計されており、処理の流れがデータの変換として読み取れることが重視されています。

副作用のないコードは、処理の結果が予測しやすく、テストもしやすくなります。また、parallelStreamのような並列処理を利用する場合でも、安全に動作するという大きなメリットがあります。Javaのラムダ式を正しく使いこなすためには、「動くかどうか」だけでなく、「安全で読みやすいか」という視点が欠かせません。

collectを使った安全で明確な書き方

記事内で紹介したcollectメソッドは、ストリームの要素をまとめて新しいコレクションとして生成するための仕組みです。Collectors.toList()を使えば、外部のリストを操作することなく、結果として必要なデータだけを取得できます。

この書き方は、副作用を避けられるだけでなく、「この処理はデータを集めている」という意図がコードから明確に読み取れる点も大きな利点です。Javaのラムダ式やStream APIを使う際には、forEachは主に表示やログ出力などの用途に限定し、データ変換や収集にはmapcollectを使う、という意識を持つことが重要です。

まとめとしてのサンプルプログラム

ここで、今回の内容を整理するために、副作用を避けた安全な書き方のサンプルプログラムを改めて確認してみましょう。記事内と同じクラス構成やコードスタイルを意識しています。


import java.util.*;
import java.util.stream.*;

public class CollectSummaryExample {
    public static void main(String[] args) {
        List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);

        List<Integer> evens = numbers.stream()
                                      .filter(n -> n % 2 == 0)
                                      .collect(Collectors.toList());

        System.out.println(evens);
    }
}

このようにcollectを使えば、処理の流れが「抽出して、集める」という形で明確になり、副作用を意識する必要もなくなります。Javaのラムダ式に慣れてきたら、この書き方を基本形として身につけておくとよいでしょう。

先生と生徒の振り返り会話

生徒

「forEachでaddするのがダメって聞いて最初は驚きましたけど、副作用って考え方が分かってきました。」

先生

「そうだね。動くコードと、良いコードは必ずしも同じじゃないという点が大事なんだ。」

生徒

「collectを使うと、何をしている処理なのかが読みやすくなりますね。」

先生

「その通り。Javaのラムダ式やStream APIは、処理の意図がコードに表れる書き方を目指すと、自然と安全な設計になるよ。」

生徒

「これからは、データを集めたいときはforEachじゃなくてcollectを使うように意識します。」

先生

「いい心がけだね。それができるようになると、Javaのラムダ式を本当の意味で使いこなせていると言えるよ。」

今回のまとめを通して、Javaのラムダ式におけるforEachaddの注意点、副作用の考え方、そして安全なcollectの使い方を整理できたはずです。Stream APIの基本を正しく理解し、読みやすく保守しやすいJavaコードを書けるようになりましょう。

Java Silver SE17(1Z0-825)の演習量を重視したい人には、 定番の問題集がこちらです。

徹底攻略 Java SE 17 Silver 問題集をAmazonで見る

※ Amazonアソシエイト・プログラムを利用しています

カテゴリの一覧へ
新着記事
PR

JavaやLinuxの検証環境に
低コストで使えるVPS

Thymeleafのth:blockの使い方を完全ガイド!初心者でもわかるテンプレートブロック管理
Thymeleafのth:selected属性の使い方を完全解説!初心者でもわかるセレクトボックス選択状態の指定方法
Spring MVCのルーティング設計をマスター!初心者向け@GetMappingと@PostMappingの基本と命名ルール
JSPのコメントタグとHTMLコメントの違いを徹底解説!初心者向けわかりやすい使い分け講座
PR 未経験からITエンジニアを目指す方へ

Javaを学んでいるけど、「このまま未経験で就職できるか不安」という20代向け。 学歴不問・無料サポートの就職支援という選択肢があります。

Tamesy |無料で面談予約
人気記事
No.1
Java&Spring記事人気No1
Spring BootとJavaの互換性一覧!3.5/3.4/3.3はJava 21・17に対応してる?
No.2
Java&Spring記事人気No2
JavaのArrayListクラスとgetメソッドを完全解説!初心者でもわかるリストの要素取得
No.3
Java&Spring記事人気No3
JavaのIntegerクラスparseIntメソッド完全ガイド!初心者でもわかる文字列から数値変換
No.4
Java&Spring記事人気No4
Spring BootのJakarta移行ガイド!初心者向けjavax→jakarta変更ポイント徹底解説
No.5
Java&Spring記事人気No5
Thymeleafのth:classappend属性の使い方を完全ガイド!初心者でもわかる動的クラス追加
No.6
Java&Spring記事人気No6
JavaのIntegerクラスの使い方を完全ガイド!初心者でもわかる整数操作
No.7
Java&Spring記事人気No7
JavaのHttpSessionを徹底解説!初心者でもわかるセッション管理の基本
No.8
Java&Spring記事人気No8
Springの@Componentアノテーションの使い方を徹底解説!初心者でもわかるSpring Boot入門
PR

ローカルPCに依存しない開発環境という選択肢

Java・Linuxの検証や学習環境を、クラウド上ですぐに用意できます。

Java入門

Javaの基礎を体系的に学びたい場合は、文法だけでなく 「なぜそう書くのか」まで丁寧に解説されているため、 初心者でも理解しやすい定番の1冊です。

スッキリわかるJava入門 第4版

※ 紙の書籍・電子書籍どちらでも購入できます

Java実践

ジェネリクス、enum、シールクラスなどの型設計から、 関数型プログラミング(ラムダ式・Stream API)、 JVM制御やリフレクション、外部ライブラリの活用までを扱っており、 「Javaを使えるレベル」へ進むための内容が網羅されています。

スッキリわかるJava入門 実践編 第4版

※ 紙の書籍・電子書籍どちらでも購入できます

Spring入門

Spring Frameworkの全体像から、 Webアプリ開発で必要となる主要機能までを 体系的に解説している定番の入門書です。

Spring徹底入門 第2版 Spring FrameworkによるJavaアプリケーション開発

※ 紙の書籍・電子書籍どちらでも購入できます

PR 実務経験のあるエンジニア向け

Javaなどの実務経験があり、次のキャリアを検討している方向け。 IT・ゲーム業界に特化した転職支援サービスという選択肢もあります。

転職ボックス |IT・ゲーム業界専門