[C言語] ファイルから一行ずつ読み込みする方法を解説

C言語でファイルから一行ずつ読み込むには、fgets関数を使用します。

この関数は、指定したファイルポインタから一行を読み込み、文字列としてバッファに格納します。

バッファのサイズを指定することで、読み込む最大文字数を制限できます。

また、fgetsは改行文字も含めて読み込むため、行の終わりを正確に把握できます。

ファイルの終端に達すると、fgetsNULLを返します。

この方法は、テキストファイルを行単位で処理する際に非常に便利です。

この記事でわかること
  • fgets関数を使用したファイルの一行読み込み方法
  • ファイル操作におけるエラー処理の方法
  • ファイル内容の逆順表示や特定文字列の抽出方法
  • 行番号を付けてファイル内容を表示する方法

目次から探す

一行ずつ読み込む方法

ファイルから一行ずつデータを読み込むことは、テキストファイルの処理において非常に重要です。

C言語では、fgets関数を使用してこの操作を簡単に行うことができます。

以下では、fgets関数の使い方や注意点について詳しく解説します。

fgets関数の使い方

fgets関数は、指定されたファイルから一行を読み込むための関数です。

以下に基本的な使い方を示します。

#include <stdio.h>
int main() {
    FILE *file;
    char buffer[256]; // 読み込むバッファ
    // ファイルを読み込みモードで開く
    file = fopen("example.txt", "r");
    if (file == NULL) {
        printf("ファイルを開けませんでした。\n");
        return 1;
    }
    // ファイルから一行ずつ読み込む
    while (fgets(buffer, sizeof(buffer), file) != NULL) {
        printf("%s", buffer); // 読み込んだ行を表示
    }
    // ファイルを閉じる
    fclose(file);
    return 0;
}

このコードは、example.txtというファイルを開き、内容を一行ずつ読み込んで表示します。

fgets関数は、ファイルの終わりに達するとNULLを返します。

これはサンプルテキストです。
複数行にわたって内容が書かれています。

この例では、example.txtの内容がそのまま出力されます。

fgetsとfgetsの違い

fgets関数は、ファイルから一行を読み込むために使用されますが、gets関数は標準入力から一行を読み込むために使用されます。

以下に両者の違いを表にまとめます。

スクロールできます
関数名用途安全性
fgetsファイルから読み込むバッファサイズを指定するため安全
gets標準入力から読み込むバッファオーバーフローの危険があるため非推奨

gets関数はバッファサイズを指定しないため、バッファオーバーフローの危険性があり、現在では使用が推奨されていません。

fgetsを使用することで、安全にデータを読み込むことができます。

バッファサイズの設定

fgets関数を使用する際には、バッファサイズを適切に設定することが重要です。

バッファサイズは、読み込む最大文字数を指定します。

以下にバッファサイズの設定に関するポイントを示します。

  • バッファサイズは、読み込む行の最大長よりも大きく設定する必要があります。
  • バッファサイズには、終端文字\0のためのスペースも含める必要があります。
  • バッファサイズが小さいと、行の一部しか読み込まれない可能性があります。

適切なバッファサイズを設定することで、ファイルからのデータ読み込みを効率的に行うことができます。

エラー処理

ファイル操作を行う際には、さまざまなエラーが発生する可能性があります。

これらのエラーを適切に処理することで、プログラムの信頼性を向上させることができます。

以下では、ファイル操作における一般的なエラーとその対処法について解説します。

ファイルが開けない場合の対処

ファイルを開く際に、指定したファイルが存在しない、またはアクセス権がない場合などにエラーが発生します。

このような場合には、fopen関数NULLを返します。

以下に対処法を示します。

#include <stdio.h>
int main() {
    FILE *file;
    // ファイルを読み込みモードで開く
    file = fopen("example.txt", "r");
    if (file == NULL) {
        perror("ファイルを開けませんでした");
        return 1;
    }
    // ファイルを閉じる
    fclose(file);
    return 0;
}

この例では、fopenが失敗した場合にperror関数を使用してエラーメッセージを表示します。

perrorは、エラーの原因を標準エラー出力に表示するために便利です。

読み込みエラーの検出

ファイルからの読み込み中にエラーが発生することがあります。

fgets関数を使用している場合、エラーが発生するとNULLが返されますが、ファイルの終わりに達した場合もNULLが返されるため、feof関数ferror関数を使用してエラーを検出します。

#include <stdio.h>
int main() {
    FILE *file;
    char buffer[256];
    file = fopen("example.txt", "r");
    if (file == NULL) {
        perror("ファイルを開けませんでした");
        return 1;
    }
    while (fgets(buffer, sizeof(buffer), file) != NULL) {
        printf("%s", buffer);
    }
    if (ferror(file)) {
        printf("読み込み中にエラーが発生しました。\n");
    }
    fclose(file);
    return 0;
}

このコードでは、ferror関数を使用して読み込み中のエラーを検出し、エラーメッセージを表示します。

エラーメッセージの表示

エラーメッセージを表示することで、ユーザーや開発者は問題の原因を特定しやすくなります。

perror関数strerror関数を使用して、エラーメッセージを表示することが一般的です。

  • perror関数: 標準エラー出力にエラーメッセージを表示します。

例:perror("エラーメッセージ")

  • strerror関数: エラー番号に対応するエラーメッセージを返します。

例:printf("%s\n", strerror(errno))

これらの関数を使用することで、エラーの詳細をユーザーに伝えることができます。

エラーメッセージを適切に表示することで、プログラムのデバッグやトラブルシューティングが容易になります。

応用例

ファイルから一行ずつ読み込む基本的な方法を理解したら、次はその応用例を見ていきましょう。

ここでは、ファイルの内容を逆順に表示する方法、特定の文字列を含む行を抽出する方法、そして行番号を付けて表示する方法について解説します。

ファイルの内容を逆順に表示

ファイルの内容を逆順に表示するには、まず全ての行をメモリに読み込んでから逆順に出力します。

以下にその方法を示します。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX_LINES 1000
#define MAX_LENGTH 256
int main() {
    FILE *file;
    char *lines[MAX_LINES];
    int count = 0;
    file = fopen("example.txt", "r");
    if (file == NULL) {
        perror("ファイルを開けませんでした");
        return 1;
    }
    // ファイルから全ての行を読み込む
    while (count < MAX_LINES && fgets(lines[count] = malloc(MAX_LENGTH), MAX_LENGTH, file) != NULL) {
        count++;
    }
    fclose(file);
    // 逆順に表示
    for (int i = count - 1; i >= 0; i--) {
        printf("%s", lines[i]);
        free(lines[i]); // メモリを解放
    }
    return 0;
}

このプログラムは、ファイルの内容をメモリに保存し、逆順に表示します。

メモリ管理に注意し、使用後は必ずfree関数で解放します。

特定の文字列を含む行の抽出

特定の文字列を含む行を抽出するには、strstr関数を使用して各行を検索します。

以下にその方法を示します。

#include <stdio.h>
#include <string.h>
int main() {
    FILE *file;
    char buffer[256];
    const char *keyword = "検索文字列";
    file = fopen("example.txt", "r");
    if (file == NULL) {
        perror("ファイルを開けませんでした");
        return 1;
    }
    // 各行を検索
    while (fgets(buffer, sizeof(buffer), file) != NULL) {
        if (strstr(buffer, keyword) != NULL) {
            printf("%s", buffer); // キーワードを含む行を表示
        }
    }
    fclose(file);
    return 0;
}

このプログラムは、example.txt内の各行を調べ、指定したキーワードを含む行を表示します。

行番号を付けて表示

ファイルの各行に行番号を付けて表示するには、単に行を読み込む際にカウンタを使用します。

以下にその方法を示します。

#include <stdio.h>
int main() {
    FILE *file;
    char buffer[256];
    int lineNumber = 1;
    file = fopen("example.txt", "r");
    if (file == NULL) {
        perror("ファイルを開けませんでした");
        return 1;
    }
    // 各行に行番号を付けて表示
    while (fgets(buffer, sizeof(buffer), file) != NULL) {
        printf("%d: %s", lineNumber++, buffer);
    }
    fclose(file);
    return 0;
}

このプログラムは、ファイルの各行に行番号を付けて表示します。

行番号は1から始まり、各行を読み込むたびにインクリメントされます。

これらの応用例を通じて、ファイル操作の幅広い可能性を理解し、実際のプログラムに応用することができます。

よくある質問

fgetsで改行が残るのはなぜ?

fgets関数は、行を読み込む際に改行文字も含めてバッファに格納します。

これは、fgetsが行の終わりを示すために改行をそのまま保持する仕様だからです。

改行を取り除きたい場合は、strcspn関数を使用して改行文字を検出し、'\0'で置き換えることができます。

例:buffer[strcspn(buffer, "\n")] = '\0';

ファイルの終わりをどう判断する?

ファイルの終わりを判断するには、fgets関数NULLを返した後にfeof関数を使用します。

feof関数は、ファイルの終端に達したかどうかを確認するために使用されます。

fgetsNULLを返した場合、feof(file)が真であればファイルの終わりに達したことを示します。

バッファサイズを超えた場合の対処法は?

fgets関数は、指定されたバッファサイズを超える行を読み込むと、バッファの最大サイズまでしか読み込みません。

この場合、行の残りの部分は次のfgets呼び出しで読み込まれます。

バッファサイズを超える行を処理するには、バッファを再度使用して残りの行を読み込むループを実装する必要があります。

まとめ

ファイルから一行ずつ読み込む方法は、C言語でのファイル操作の基本的なスキルです。

この記事では、fgets関数の使い方やエラー処理、応用例について詳しく解説しました。

これらの知識を活用して、より複雑なファイル操作を行うことができるようになります。

ぜひ、実際のプログラムでこれらのテクニックを試してみてください。

  • URLをコピーしました!
目次から探す