読み込み中...

C言語のtime.hマスターへの道!初心者でもわかる5つのステップ

C言語のtime.hライブラリーの詳細な使い方とサンプルコードを含む解説記事の表紙 C言語
この記事は約13分で読めます。

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

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

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

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

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

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

はじめに

皆さん、こんにちは。

C言語について学習中の方や、これからC言語を学ぼうと考えている方に向けて、C言語の重要なライブラリの一つであるtime.hについて解説します。

この記事では、time.hの基本的な関数から、具体的なサンプルコードを交えての解説、さらに注意点や応用例についても触れていきます。

初心者の方でも理解しやすいよう、分かりやすい言葉で説明していきますので、安心して読み進めてください。

●time.hとは

C言語のtime.hは、時間や日付を扱うための標準ライブラリであり、時間の取得、変換、計算など、さまざまな時間関連の操作をサポートしています。

time.hはヘッダーファイルの一つで、これをインクルードすることにより、その機能をプログラム内で使用することができます。

●time.hの基本的な関数

time.hには様々な関数が用意されていますが、ここでは主要な5つの関数を紹介します。

○time関数

time関数は、現在のシステム時刻を取得する関数です。

この関数を使うと、現在の日付や時間を取得することができます。

#include <time.h>
#include <stdio.h>

int main() {
    time_t current_time;
    current_time = time(NULL);
    printf("%ld\n", current_time);
    return 0;
}

このコードでは、まずtime_t型の変数current_timeを宣言し、そこにtime関数を用いて現在のシステム時刻を取得しています。

time関数の引数にNULLを渡すと、現在のシステム時刻が取得できます。そしてprintf関数でその値を出力しています。

このコードを実行すると、1970年1月1日00:00:00(UNIXエポック)からの経過秒数(エポックタイム)が出力されます。

○ctime関数

ctime関数は、time関数で取得したエポックタイムを、わかりやすい形式(年-月-日 時:分:秒 改行)に変換する関数です。

#include <time.h>
#include <stdio.h>

int main() {
    time_t current_time;
    current_time = time(NULL);
    printf("%s", ctime(&current_time));
    return 0;
}

このコードでは、まず現在のエポックタイムを取得し、その値をctime関数に渡しています。

そしてprintf関数でその変換結果を出力しています。

このコードを実行すると、現在の日時が「年-月-日 時:分:秒 改行」の形式で出力されます。

○difftime関数

次に紹介するdifftime関数は、二つの時間の差を秒単位で計算する関数です。

この関数を使うと、特定の時間間隔を計算することができます。

#include <time.h>
#include <stdio.h>

int main() {
    time_t start_time, end_time;
    double elapsed_time;

    start_time = time(NULL);
    // 時間が経過する処理
    for(int i = 0; i < 10000000; i++);
    end_time = time(NULL);

    elapsed_time = difftime(end_time, start_time);
    printf("Elapsed time: %.2f seconds\n", elapsed_time);
    return 0;
}

このコードでは、まず開始時間を取得し、その後に時間が経過する処理(ここでは無意味なforループ)を行ってから終了時間を取得しています。

そしてdifftime関数を用いて、その二つの時間の差(経過時間)を計算し、その結果を出力しています。

このコードを実行すると、処理にかかった時間が秒単位で出力されます。

○mktime関数

mktime関数は、構造体tmの情報を元にエポックタイムを生成する関数です。

構造体tmは、時間情報を詳細に扱うための構造体で、年、月、日、時、分、秒などの情報を持つことができます。

#include <time.h>
#include <stdio.h>

int main() {
    struct tm t;
    time_t epoch_time;

    t.tm_year = 2023-1900;
    t.tm_mon = 7-1;
    t.tm_mday = 24;
    t.tm_hour = 12;
    t.tm_min = 0;
    t.tm_sec = 0;
    t.tm_isdst = -1;

    epoch_time = mktime(&t);
    printf("%ld\n", epoch_time);
    return 0;
}

このコードでは、まず構造体tmを用いて時間情報を指定しています。

tm_yearには年を1900減算した値、tm_monには月を1減算した値を設定する必要があります。

そしてmktime関数にその構造体tmのアドレスを渡すことで、エポックタイムを生成しています。

このコードを実行すると、指定した日時のエポックタイムが出力されます。

○localtime関数

最後に紹介するlocaltime関数は、エポックタイムを構造体tmに変換する関数です。

この関数を使うと、エポックタイムから詳細な時間情報を取得することができます。

#include <time.h>
#include <stdio.h>

int main() {
    time_t current_time;
    struct tm *t;

    current_time = time(NULL);
    t = localtime(&current_time);

    printf("%d年%d月%d日 %d時%d分%d秒\n",
           t->tm_year + 1900,
           t->tm_mon + 1,
           t->tm_mday,
           t->tm_hour,
           t->tm_min,
           t->tm_sec);
    return 0;
}

このコードでは、まず現在のエポックタイムを取得し、その値をlocaltime関数に渡しています。

そしてprintf関数でその変換結果を出力しています。

このコードを実行すると、現在の日時が「年月日 時分秒」の形式で出力されます。

これらの基本的な関数を使いこなすことで、C言語で時間に関する様々な操作を行うことができます。

続いて、これらの関数を使ったサンプルコードを紹介します。

●time.hを使ったサンプルコード

○現在時刻の取得

最初に紹介するサンプルコードは、現在の日時を取得するコードです。

このコードでは、先ほど説明したtime関数とlocaltime関数を使用しています。

#include <time.h>
#include <stdio.h>

int main() {
    time_t current_time;
    struct tm *t;

    current_time = time(NULL);
    t = localtime(&current_time);

    printf("現在の日時: %d年%d月%d日 %d時%d分%d秒\n",
           t->tm_year + 1900,
           t->tm_mon + 1,
           t->tm_mday,
           t->tm_hour,
           t->tm_min,
           t->tm_sec);
    return 0;
}

このコードを実行すると、現在の日時が出力されます。

現在の日付や時間を取得することは、ログの記録やエラー発生時の時間記録など、様々な場面で役立ちます。

○特定の日付・時間への変換

次に紹介するサンプルコードは、指定した日付・時間へエポックタイムを変換するコードです。

このコードでは、mktime関数とctime関数を使用しています。

#include <time.h>
#include <stdio.h>

int main() {
    struct tm t;
    time_t epoch_time;

    t.tm_year = 2023-1900;
    t.tm_mon = 7-1;
    t.tm_mday = 24;
    t.tm_hour = 12;
    t.tm_min = 0;
    t.tm_sec = 0;
    t.tm_isdst = -1;

    epoch_time = mktime(&t);
    printf("指定日時: %s", ctime(&epoch_time));
    return 0;
}

このコードを実行すると、指定した日時が出力されます。

このような日付・時間の変換は、特定の日時を基準にした処理や、日時のフォーマット変換などで利用できます。

○二つの時間の差を計算

最後に紹介するサンプルコードは、二つの時間の差を計算するコードです。

このコードでは、difftime関数を使用しています。

#include <time.h>
#include <stdio.h>

int main() {
    time_t start_time, end_time;
    double elapsed_time;

    start_time = time(NULL);
    // 時間が経過する処理
    for(int i = 0; i < 10000000; i++);
    end_time = time(NULL);

    elapsed_time = difftime(end_time, start_time);
    printf("経過時間: %.2f秒\n", elapsed_time);
    return 0;
}

このコードを実行すると、処理にかかった時間が秒単位で出力されます。

このような時間の差の計算は、プログラムのパフォーマンス計測や、一定時間待機する処理などで利用できます。

これらのサンプルコードは非常に基本的な使用例ですが、これらの関数を組み合わせることで、さまざまな時間関連の操作を行うことができます。

●time.hの利用時の注意点と対策

time.hを利用する際の一つの注意点として、time_t型の範囲について説明します。

time_t型は通常、long型として定義されており、その範囲はシステムによりますが、通常は約1901年から約2038年までの時間を表現できます。

これは、32ビット整数としてエポックタイムを表現した場合の範囲です。

したがって、2038年以降の時間を扱う必要がある場合、64ビット整数を使用するシステムを使用することを検討してください。

これはいわゆる「2038年問題」と呼ばれるもので、既存のソフトウェアやシステムが2038年以降も正常に動作するように、早めの対策が必要とされています。

また、time.hの関数は、一部を除きスレッドセーフではありません。

したがって、マルチスレッドプログラムでこれらの関数を使用する場合は、排他制御を適切に行う必要があります。

特に、ctime関数やlocaltime関数は、内部的に静的なメモリ領域を使用するため、スレッドセーフではない関数を安全に使用するためには、それぞれctime_r関数やlocaltime_r関数を使用することが推奨されています。

●time.hの応用例とサンプルコード

○日付と時間のカウントダウンタイマー

time.hを使った応用例として、日付と時間のカウントダウンタイマーを作ることができます。

#include <time.h>
#include <stdio.h>

int main() {
    time_t current_time, target_time;
    struct tm t;
    double diff_seconds;

    // カウントダウンターゲットの日時を設定
    t.tm_year = 2023-1900;
    t.tm_mon = 12-1;
    t.tm_mday = 31;
    t.tm_hour = 23;
    t.tm_min = 59;
    t.tm_sec = 59;
    t.tm_isdst = -1;
    target_time = mktime(&t);

    while (1) {
        current_time = time(NULL);
        diff_seconds = difftime(target_time, current_time);

        if (diff_seconds <= 0.0) {
            printf("カウントダウン終了\n");
            break;
        }

        printf("残り: %ld日 %ld時間 %ld分 %ld秒\n",
               (long)diff_seconds / (24 * 60 * 60),
               (long)diff_seconds / (60 * 60) % 24,
               (long)diff_seconds / 60 % 60,
               (long)diff_seconds % 60);
        sleep(1);  // 1秒待機
    }

    return 0;
}

このコードを実行すると、指定した日時までの残り時間がリアルタイムで表示されます。

カウントダウンが終了すると、「カウントダウン終了」と表示されます。

○プログラムの実行時間計測

また、time.hを使ってプログラムの実行時間を計測することもできます。

#include <time.h>
#include <stdio.h>

void some_long_operation() {
    // 長い時間がかかる処理
    for(int i = 0; i < 100000000; i++);
}

int main() {
    time_t start_time, end_time;
    double elapsed_time;

    start_time = time(NULL);
    some_long_operation();
    end_time = time(NULL);

    elapsed_time = difftime(end_time, start_time);
    printf("処理時間: %.2f秒\n", elapsed_time);

    return 0;
}

このコードを実行すると、関数some_long_operationの実行にかかった時間が秒単位で出力されます。

これは、プログラムのパフォーマンスを計測する際に役立つ手法です。

まとめ

この記事では、C言語で時間や日付を扱うためのライブラリーであるtime.hについて、その基本的な使い方から応用例までを紹介しました。

time.hをマスターすることで、日付や時間を扱うプログラムを効率良く書くことができます。

ただし、time.hの関数を使用する際は、スレッドセーフでないことや2038年問題など、いくつか注意点があります。

これらの点を理解し、適切な使い方をすることが重要です。

これらの基本的な使い方や応用例を参考に、C言語の時間処理をより理解し、活用してみてください。