カテゴリ: Spring 更新日: 2026/01/09

Spring Bootの大容量レスポンス配信!初心者でもわかるストリーミングとファイル出力の実装方法

Spring Bootの大容量レスポンス配信:ストリーミング/ファイル出力のコツ
Spring Bootの大容量レスポンス配信:ストリーミング/ファイル出力のコツ

教材紹介 Spring Boot 学習のおすすめ教材

Spring Bootを使ったWebアプリケーション開発を、 環境構築から実践まで一通り学びたい方には、 定評のある入門書が参考になります。

Spring Boot 3 プログラミング入門をAmazonで見る

※ Amazon広告リンク

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

生徒

「大きなファイルをSpring Bootでダウンロードさせたいんですけど、メモリが心配です…」

先生

「それはよくある悩みですね。Spring Bootではストリーミングでレスポンスを返すことで、メモリ消費を抑えながら大容量ファイルを送信できますよ。」

生徒

「普通にResponseEntityで返すだけじゃだめなんですか?」

先生

「用途によりますが、大容量のファイルを一気にメモリに載せるのは危険です。ストリームを使えば、効率よく安全に送信できます。詳しく説明しますね!」

1. Spring Bootで大容量レスポンスを扱うときの注意点

1. Spring Bootで大容量レスポンスを扱うときの注意点
1. Spring Bootで大容量レスポンスを扱うときの注意点

Spring Bootでは、通常のResponseEntity@ResponseBodyを使ってレスポンスを返すことができますが、対象が数百MBや数GBに及ぶような大容量ファイルの場合、すべてをメモリに読み込むのは非常に危険です。

そのようなケースでは、ストリーミングによって逐次的にデータを返す方法が効果的です。ストリーミングは、サーバーのメモリ使用量を最小限に抑えつつ、クライアントへリアルタイムにデータを送信できます。

2. Spring Bootでファイルをストリーミングする実装方法

2. Spring Bootでファイルをストリーミングする実装方法
2. Spring Bootでファイルをストリーミングする実装方法

以下のように、OutputStreamを使用して、ファイルを少しずつクライアントに送り出す形で処理を実装できます。これは大容量ファイルでも安定してダウンロードさせる方法です。


@GetMapping("/download")
public void downloadLargeFile(HttpServletResponse response) throws IOException {
    response.setContentType("application/octet-stream");
    response.setHeader("Content-Disposition", "attachment; filename=large-file.zip");

    try (InputStream inputStream = new FileInputStream("/path/to/large-file.zip");
         OutputStream out = response.getOutputStream()) {

        byte[] buffer = new byte[8192];
        int bytesRead;
        while ((bytesRead = inputStream.read(buffer)) != -1) {
            out.write(buffer, 0, bytesRead);
        }
        out.flush();
    }
}

この実装ではInputStreamを使ってファイルを読み込み、バッファ単位で書き出すことで、大容量でも安定したレスポンスを実現できます。

3. ファイルの種類に応じたContent-Typeの設定

3. ファイルの種類に応じたContent-Typeの設定
3. ファイルの種類に応じたContent-Typeの設定

ファイルの種類によってContent-Typeを適切に設定することで、ブラウザでの動作や表示形式を制御できます。たとえば、PDFなら「application/pdf」、CSVなら「text/csv」などです。


response.setContentType("application/pdf"); // PDFファイルの例

適切なContent-Typeを設定することで、ユーザーにとってより自然なファイルの受け取りが可能になります。

Spring FrameworkやThymeleafを使った Webアプリ開発の全体像をやさしく理解したい人には、 この入門書が定番です。

Spring Framework超入門をAmazonで見る

※ Amazon広告リンク

4. SpringのStreamingResponseBodyを使ったストリーミング配信

4. Springの<code>StreamingResponseBody</code>を使ったストリーミング配信
4. SpringのStreamingResponseBodyを使ったストリーミング配信

StreamingResponseBodyを使えば、Spring Bootが用意する仕組みで簡単にストリーミング配信を実装できます。


@GetMapping("/stream")
public ResponseEntity<StreamingResponseBody> streamLargeFile() {
    File file = new File("/path/to/large-file.mp4");

    StreamingResponseBody stream = outputStream -> {
        try (InputStream inputStream = new FileInputStream(file)) {
            byte[] buffer = new byte[8192];
            int bytesRead;
            while ((bytesRead = inputStream.read(buffer)) != -1) {
                outputStream.write(buffer, 0, bytesRead);
            }
            outputStream.flush();
        }
    };

    return ResponseEntity.ok()
        .header(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=large-file.mp4")
        .contentType(MediaType.APPLICATION_OCTET_STREAM)
        .body(stream);
}

StreamingResponseBodyは非同期処理にも対応しており、バックグラウンドでの大容量配信にも適しています。

5. ストリーミングを使うときの注意点と制限

5. ストリーミングを使うときの注意点と制限
5. ストリーミングを使うときの注意点と制限

ストリーミング配信は非常に便利ですが、以下の点に注意が必要です。

  • クライアントが途中で接続を切った場合、例外が発生する可能性があります
  • トランザクション管理と併用する場合、早めにコミットする設計が必要
  • 圧縮処理などを挟む場合は、バッファのサイズや処理速度に注意

6. ファイル出力時のセキュリティ対策とベストプラクティス

6. ファイル出力時のセキュリティ対策とベストプラクティス
6. ファイル出力時のセキュリティ対策とベストプラクティス

ユーザーが任意のファイルパスを指定できるようなAPI設計は危険です。パス・トラバーサル攻撃を防ぐために、ファイルパスは固定または検証されたものに限定すべきです。

また、以下のようなチェックを実装すると安全性が高まります。

  • 存在しないファイルのリクエストには404で返す
  • 拡張子を限定し、不正なファイル出力を避ける
  • ファイル名に../が含まれていないかチェック

7. ファイルダウンロードのログや監査も忘れずに

7. ファイルダウンロードのログや監査も忘れずに
7. ファイルダウンロードのログや監査も忘れずに

大容量ファイルを扱うAPIでは、セキュリティや監査対応のために、誰がいつ何をダウンロードしたかのログを残すことが重要です。

特に企業システムでは、情報漏洩や不正アクセスのリスク管理のため、アクセスログの出力やユーザー認証・認可とセットでダウンロード処理を設計することが推奨されます。

8. Spring Boot + セキュリティで認証付きダウンロードにするには?

8. Spring Boot + セキュリティで認証付きダウンロードにするには?
8. Spring Boot + セキュリティで認証付きダウンロードにするには?

@PreAuthorizeSecurityContextHolderを用いることで、認証されたユーザーのみがファイルを取得できるように制御することが可能です。


@PreAuthorize("hasRole('USER')")
@GetMapping("/secure-download")
public void secureDownload(HttpServletResponse response) {
    // ストリーミング処理
}

これにより、不正アクセスを防ぎ、ファイルごとのアクセス制限を実現できます。

まとめ

まとめ
まとめ

大容量ファイル配信で押さえておきたいポイントの振り返り

本記事では、Spring Bootで大容量ファイルや大きなレスポンスデータを安全かつ効率的に配信するための考え方と実装方法を学びました。 通常のResponseEntity@ResponseBodyでの返却は、小さなデータであれば問題ありませんが、 数百メガバイト以上のファイルを扱う場合、サーバーのメモリを大量に消費してしまう危険があります。 そのため、ストリーミング配信という仕組みを理解し、必要に応じて使い分けることが重要になります。

特にInputStreamOutputStreamを使った基本的なストリーミング処理は、 ファイルを少しずつ読み込みながらクライアントへ送信できるため、 メモリ使用量を最小限に抑えつつ安定したダウンロード処理を実現できます。 さらに、Springが提供するStreamingResponseBodyを使えば、 フレームワークに沿った形で、よりシンプルに非同期対応のストリーミング処理を実装できる点も大きなメリットです。

ストリーミング実装を活かすための実践的な考え方

ストリーミング配信では、単にコードを書くだけでなく、Content-Typeの設定やファイル名の指定、 クライアント側の動作も含めて設計する必要があります。 適切なContent-Typeを設定することで、ブラウザ上での挙動が自然になり、 ユーザーにとって使いやすいファイルダウンロード体験を提供できます。

また、セキュリティ面も非常に重要です。 任意のパスをそのまま受け取ってファイルを出力する設計は避け、 出力可能なファイルやディレクトリを制限することで、 不正アクセスや情報漏洩のリスクを下げることができます。 企業向けシステムや業務アプリケーションでは、 ダウンロードログや監査ログを残す設計も欠かせません。

まとめ用の簡単なサンプルコード

ここで、記事全体の内容を踏まえた、シンプルなストリーミング配信のイメージを再確認してみましょう。 以下は、大容量ファイルを少しずつクライアントに送信する基本形です。


@GetMapping("/summary-download")
public void summaryDownload(HttpServletResponse response) throws IOException {
    response.setContentType("application/octet-stream");
    response.setHeader("Content-Disposition", "attachment; filename=sample.zip");

    try (InputStream in = new FileInputStream("/path/to/sample.zip");
         OutputStream out = response.getOutputStream()) {

        byte[] buffer = new byte[8192];
        int len;
        while ((len = in.read(buffer)) != -1) {
            out.write(buffer, 0, len);
        }
        out.flush();
    }
}

このような形をベースに、認証・認可、ログ出力、例外処理を組み合わせていくことで、 実務でも安心して使えるファイル配信機能を構築できます。

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

生徒

「Spring Bootで大きなファイルを扱うときは、普通に返すんじゃなくて、 ストリーミングを使う理由がよく分かりました。 メモリを節約できるのが大きいですね。」

先生

「その通りです。特に大容量ファイルや動画、バックアップデータなどを扱う場合は、 ストリーミング配信がほぼ必須になります。 安定性と安全性の両方を意識することが大切ですね。」

生徒

StreamingResponseBodyも便利そうですね。 実装がスッキリして、Springらしい書き方だと感じました。」

先生

「ええ。最初は基本のInputStreamOutputStreamを理解して、 その後でStreamingResponseBodyを使うと理解が深まります。 実務ではセキュリティやログも忘れずに設計してくださいね。」

カテゴリの一覧へ
新着記事
JavaのIndexOutOfBoundsExceptionを完全ガイド!初心者でも理解できる例外処理と回避方法
スッキリわかるJava入門 第4版|独学でもレッスンでも学べる完全ガイド
JavaのCookieクラスのgetCommentメソッドを解説!Servlet開発でクッキーの説明文を取得する方法
JavaのBufferedOutputStreamクラスのwriteメソッドを徹底解説!初心者でもわかるファイルへのバイナリ書き込み
人気記事
No.1
Java&Spring記事人気No1
Spring BootとJavaの互換性一覧!3.5/3.4/3.3はJava 21・17に対応してる?
No.2
Java&Spring記事人気No2
Springの@Serviceアノテーションの使い方を徹底解説!初心者でもわかるSpring フレームワーク入門
No.3
Java&Spring記事人気No3
Thymeleafのth:classappend属性の使い方を完全ガイド!初心者でもわかる動的クラス追加
No.4
Java&Spring記事人気No4
Spring Data JPA入門!findAll()やfindBy**()の使い方などデータベース操作の基礎を学ぶ
No.5
Java&Spring記事人気No5
Spring BootのJakarta移行ガイド!初心者向けjavax→jakarta変更ポイント徹底解説
No.6
Java&Spring記事人気No6
Thymeleaf(タイムリーフ)入門!初心者でもわかるSpring Bootとテンプレートエンジンの使い方
No.7
Java&Spring記事人気No7
Springの@Repositoryアノテーションの使い方を徹底解説!初心者でもわかるSpringフレームワークのデータアクセス
No.8
Java&Spring記事人気No8
SpringのPageableとSortでページング一覧APIを最短実装!初心者向け完全ガイド

💻 作業効率アップに

長時間のコーディングでも疲れにくい♪ 静音ワイヤレスマウス

Logicool Signature M750 を見る

※ Amazon広告リンク