カテゴリ: C# 更新日: 2026/03/03

C#で月末・月初を計算する方法(AddMonths・DaysInMonth)完全ガイド

C#で月末・月初を計算する方法(AddMonths・DaysInMonth)
C#で月末・月初を計算する方法(AddMonths・DaysInMonth)

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

生徒

「プログラムで今月の月末や来月の月初を自動で求めたいです。簡単にできますか?」

先生

「はい。C#には日付操作に便利な機能が用意されています。特にDateTime構造体とDateTime.DaysInMonthDateTime.AddMonthsを組み合わせると簡単に月初・月末を計算できます。」

生徒

「具体例を見せてください。プログラミング未経験でもわかるように説明してください。」

先生

「いいですよ。まずは用語の簡単な説明から始めますね。」

1. 用語のやさしい説明(初心者向け)

1. 用語のやさしい説明(初心者向け)
1. 用語のやさしい説明(初心者向け)

DateTime(デートタイム)は、年・月・日・時・分・秒などの「日時」を表す道具です。AddMonthsはその日時に指定した月数を足す(または引く)ための機能、DaysInMonthは指定した年と月に何日あるかを教えてくれる関数です。たとえば「2024年2月は29日あるか?」という疑問に答えてくれます。

2. 考え方(かんたんな例え)

2. 考え方(かんたんな例え)
2. 考え方(かんたんな例え)

月初・月末を求める作業はカレンダーのページめくりに似ています。今持っている日付を基準に「その月の最初の日」を取得するのが月初、「その月の最後の日」を取得するのが月末です。DaysInMonthでその月の日数を調べ、日付の「日」だけを差し替えればOKです。

3. サンプルコード:月初と月末を取得する基本パターン

3. サンプルコード:月初と月末を取得する基本パターン
3. サンプルコード:月初と月末を取得する基本パターン

以下は、指定した日付からその月の月初と月末を返すシンプルな関数例です。コード中のコメントも読んでください。


using System;

class Program
{
    // 指定日付の月初を返す
    static DateTime GetFirstDayOfMonth(DateTime date)
    {
        return new DateTime(date.Year, date.Month, 1);
    }

    // 指定日付の月末を返す
    static DateTime GetLastDayOfMonth(DateTime date)
    {
        int days = DateTime.DaysInMonth(date.Year, date.Month);
        return new DateTime(date.Year, date.Month, days);
    }

    static void Main()
    {
        DateTime today = DateTime.Now;
        DateTime first = GetFirstDayOfMonth(today);
        DateTime last = GetLastDayOfMonth(today);

        Console.WriteLine($"今日: {today}");
        Console.WriteLine($"月初: {first}");
        Console.WriteLine($"月末: {last}");
    }
}

(ここに出力結果)

この例ではDateTime.DaysInMonth(year, month)でその月の日数を得て、月末の日付を作成しています。月初は単純に1日を指定します。

4. AddMonthsを使った応用例:翌月の月初・月末

4. AddMonthsを使った応用例:翌月の月初・月末
4. AddMonthsを使った応用例:翌月の月初・月末

「来月の月初」や「先月の月末」を簡単に求めたいときは、AddMonths(1)AddMonths(-1)が便利です。AddMonthsは月をまたいだ補正(たとえば1月31日に1か月足すと3月3日になるような挙動)を内部で扱いますが、月初や月末を求めるときは基準日を月の1日に変えてからAddMonthsすると安全です。


// 来月の月初を求める例
DateTime thisMonthFirst = new DateTime(today.Year, today.Month, 1);
DateTime nextMonthFirst = thisMonthFirst.AddMonths(1);

// 来月の月末を求める例
int nextMonthDays = DateTime.DaysInMonth(nextMonthFirst.Year, nextMonthFirst.Month);
DateTime nextMonthLast = new DateTime(nextMonthFirst.Year, nextMonthFirst.Month, nextMonthDays);

この流れだと、1月31日のような「月によって日数が違う日」でも安全に翌月や前月を計算できます。

5. 実践的な注意点とよくある疑問

5. 実践的な注意点とよくある疑問
5. 実践的な注意点とよくある疑問
  • タイムゾーンDateTime.Nowは実行環境のローカル時間を返します。UTC基準で扱いたければDateTime.UtcNowを使います。
  • AddMonthsの端数処理:1月31日にAddMonths(1)すると2月に同じ日は存在しないため最終日に自動で調整されます。意図した日付になるか確認しましょう。
  • 年またぎ:12月から翌年1月になる場合もAddMonthsDaysInMonthの組み合わせで正しく計算できます。

6. よく使うキーワード(SEO向け)

6. よく使うキーワード(SEO向け)
6. よく使うキーワード(SEO向け)

C# 月末取得, C# 月初取得, DateTime 月末, DateTime.DaysInMonth, AddMonths 使い方, C# 日付計算, 月の最終日, 月の初日, .NET 日付操作

7. まとめ代わりのワンポイント(補足)

7. まとめ代わりのワンポイント(補足)
7. まとめ代わりのワンポイント(補足)

基本は「基準日を月の1日にしてからAddMonthsで移動」+「DaysInMonthで日数を取得して月末を作る」という2ステップです。これだけ覚えれば月初・月末の計算はほとんどのケースで困りません。

まとめ

まとめ
まとめ

これまでの内容を振り返ると、C#における日付操作、特に「月初」と「月末」を正確に求める方法は、システム開発において欠かせないスキルであることがわかります。業務システムやWebアプリケーションでは、請求処理の締め日計算、月次レポートの集計範囲指定、スケジュール管理機能など、至る所でこのロジックが登場します。

基本となるのは、DateTime構造体の特性を理解することです。特に、DateTime.DaysInMonthメソッドは非常に強力で、閏年(うるうどし)のような複雑な計算もバックグラウンドで自動的に処理してくれます。これにより、開発者がわざわざ「2月が28日か29日か」を判定するIF文を書く必要がなくなります。

C#による日付計算のベストプラクティス

実務でより安全に日付を扱うためのポイントをいくつか深掘りしてみましょう。まず、月末を求める際、多くの方が「翌月の1日から1日引く」という手法を思い浮かべるかもしれません。確かにその方法でも月末は取得できますが、コードの可読性(読みやすさ)を考慮すると、DaysInMonthを使って直接その月の日数を取得し、日付オブジェクトを生成する方が、意図が明確に伝わります。

また、AddMonthsメソッドを使用する際の挙動には注意が必要です。たとえば、1月31日にAddMonths(1)を適用すると、2月には31日が存在しないため、自動的に2月の最終日(28日または29日)に調整されます。この暗黙的な調整を避けるためには、一度その月の「1日」にリセットしてから月を加算し、その後に再度月末を計算し直すという手順を踏むのが、最もバグの少ない堅実な実装方法と言えます。

さらに応用:COBOLからC#への移行を考える

古い基幹システム(メインフレーム)などで使われているCOBOLからC#へリプレースするプロジェクトも多いでしょう。COBOLでは日付を文字列や数値として扱い、自作のロジックで月末判定を行うことが一般的でしたが、C#では標準ライブラリだけで完結します。

参考までに、同様の処理をCOBOLで行う場合と、C#でより詳細に実装する場合の比較を見てみましょう。

【参考】COBOLでの月末算出イメージ


       IDENTIFICATION DIVISION.
       PROGRAM-ID. GET-LAST-DAY.
       DATA DIVISION.
       WORKING-STORAGE SECTION.
       01  W-DATE.
           05  W-YEAR  PIC 9(04).
           05  W-MONTH PIC 9(02).
           05  W-DAY   PIC 9(02).
       01  W-LAST-DAY  PIC 9(02).

       PROCEDURE DIVISION.
           * ここでは2024年2月を例にします
           MOVE 2024 TO W-YEAR.
           MOVE 02 TO W-MONTH.
           
           * 月によって日数を判定するロジックが必要(簡略化)
           EVALUATE W-MONTH
               WHEN 01 WHEN 03 WHEN 05 WHEN 07 WHEN 08 WHEN 10 WHEN 12
                   MOVE 31 TO W-LAST-DAY
               WHEN 04 WHEN 06 WHEN 09 WHEN 11
                   MOVE 30 TO W-LAST-DAY
               WHEN 02
                   * うるう年判定ロジックが必要
                   MOVE 29 TO W-LAST-DAY
           END-EVALUATE.

           DISPLAY "LAST DAY IS: " W-LAST-DAY.
           STOP RUN.

【決定版】C#での汎用的な日付計算クラス

C#では、これらの判定をすべてDateTimeが肩代わりしてくれます。以下に、再利用可能なユーティリティクラスとしての実装例を示します。


using System;

namespace DateUtilityApp
{
    public static class DateHelper
    {
        /// <summary>
        /// 指定した日付の月の初日(月初)を取得します。
        /// </summary>
        public static DateTime GetFirstDay(DateTime targetDate)
        {
            return new DateTime(targetDate.Year, targetDate.Month, 1);
        }

        /// <summary>
        /// 指定した日付の月の最終日(月末)を取得します。
        /// </summary>
        public static DateTime GetLastDay(DateTime targetDate)
        {
            int lastDay = DateTime.DaysInMonth(targetDate.Year, targetDate.Month);
            return new DateTime(targetDate.Year, targetDate.Month, lastDay);
        }

        /// <summary>
        /// 指定した月の「nヶ月後」の月末を取得します。
        /// </summary>
        public static DateTime GetLastDayOfRelativeMonth(DateTime targetDate, int monthsToAdd)
        {
            // まず月初に移動してから月を加算することで、端数トラブルを防ぐ
            DateTime firstDayOfNextMonth = GetFirstDay(targetDate).AddMonths(monthsToAdd);
            return GetLastDay(firstDayOfNextMonth);
        }
    }

    class Program
    {
        static void Main()
        {
            // 2024年2月(うるう年)の日付を基準にする
            DateTime baseDate = new DateTime(2024, 2, 10);
            
            DateTime firstDay = DateHelper.GetFirstDay(baseDate);
            DateTime lastDay = DateHelper.GetLastDay(baseDate);
            DateTime nextMonthLastDay = DateHelper.GetLastDayOfRelativeMonth(baseDate, 1);

            Console.WriteLine($"基準日: {baseDate:yyyy/MM/dd}");
            Console.WriteLine($"月初日: {firstDay:yyyy/MM/dd}");
            Console.WriteLine($"月末日: {lastDay:yyyy/MM/dd}");
            Console.WriteLine($"来月末: {nextMonthLastDay:yyyy/MM/dd}");
        }
    }
}

基準日: 2024/02/10
月初日: 2024/02/01
月末日: 2024/02/29
来月末: 2024/03/31

このように、C#の機能を正しく使えば、コードは非常に短く、かつ意図が分かりやすくなります。DateTimeは値型であるため、メソッドに渡しても元の値が書き換わる心配がない(イミュータブルな性質を持つ)点も、安全なプログラミングには欠かせない要素です。

検索エンジンで「C# 日付計算」と調べる際、多くのサイトが古い書き方を紹介していることもありますが、現代的な.NET環境(.NET 6/7/8以降など)でも、このDateTimeを用いた手法は鉄板の基本です。もし時間まで厳密に管理したい場合は、.Dateプロパティを使って時刻部分を切り捨てる(00:00:00にする)癖をつけておくと、比較演算での予期せぬミスを防げます。

この記事を通して、C#での日付操作に対する苦手意識が少しでも解消されれば幸いです。プログラミングは「いかに楽をして、正確な結果を得るか」が重要です。用意されている便利なライブラリを最大限に活用していきましょう。

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

生徒

「先生、ありがとうございました!DateTime.DaysInMonthを使うだけで、うるう年まで勝手に計算してくれるなんて驚きです。以前自分で計算ロジックを作ろうとして、頭がこんがらがったことがあったので……。」

先生

「ふふふ、そうでしょう。プログラミング言語の標準機能を使いこなすのは、効率的な開発の第一歩ですよ。自分でロジックを書くとどうしてもバグが入り込みやすいですが、公式が用意したライブラリなら信頼性が高いですからね。」

生徒

「まとめのコードにあった、一度月初に移動してからAddMonthsするっていうテクニックも目から鱗でした。1月31日に1ヶ月足すとどうなるか、今まで意識したことがなかったです。」

先生

「そこは中級者でもうっかりミスをしやすいポイントです。日付の足し算引き算をするときは、『基準をどこに置くか』を明確にすることが大切ですよ。来月の給与計算や、サブスクリプションの更新日計算など、実務では非常に重宝する考え方です。」

生徒

「COBOLの例も興味深かったです。昔の言語だとあんなに大変だったんですね……。C#だと数行で済むので、ますますC#が好きになりました。次は『特定の営業日を求める方法』とかも知りたくなってきました!」

先生

「いい意気込みですね!営業日の計算には祝日の考慮が必要なので、また少し工夫が必要ですが、基本のDateTimeをマスターした君ならすぐに習得できますよ。頑張りましょう!」

カテゴリの一覧へ
新着記事
New1
C#
C#でタイムスタンプの精度を上げるためのテクニックを徹底解説!(初心者向け)
New2
COBOL
COBOL資産のバージョン管理とは?初心者でもわかる保守と管理の基本
New3
C#
C#でJSONを部分的に読み込み・書き換え!初心者向け操作ガイド
New4
C#
C#のif文の使い方を完全ガイド!初心者でもわかる乱数生成
人気記事
No.1
Java&Spring記事人気No1
C#
C#のLINQでFirstとFirstOrDefaultの違いと使い方を完全解説!初心者向けガイド
No.2
Java&Spring記事人気No2
C#
C#のpartialクラスとは?初心者でも理解できるクラス分割の基本
No.3
Java&Spring記事人気No3
C#
C#のstaticクラスとstaticメソッドの基本と使い方をやさしく解説!
No.4
Java&Spring記事人気No4
COBOL
COBOLの数値データ型「PIC 9」の使い方と注意点をやさしく解説!
No.5
Java&Spring記事人気No5
C#
C#で型を調べる方法!GetType()・typeof演算子の違いと使い方
No.6
Java&Spring記事人気No6
C#
C#のCancellationTokenを使ったキャンセル処理を完全ガイド!非同期処理を安全に止める方法
No.7
Java&Spring記事人気No7
COBOL
COBOLのCOPY句の使い方を完全ガイド!初心者でもわかる共通部品の再利用方法
No.8
Java&Spring記事人気No8
C#
C#の引数と戻り値の基本!値を受け渡し・返す仕組みを理解しよう