特定の日付がある月の最初の日と最後の日を取得したい。日付は、UIフィールドの値から取得されます。
タイムピッカーを使用している場合は、
var maxDay = dtpAttendance.MaxDate.Day;
しかし、DateTimeオブジェクトから取得しようとしています。だからこれがあれば...
DateTime dt = DateTime.today;
月の初日と末日を取得する方法はdt
?
特定の日付がある月の最初の日と最後の日を取得したい。日付は、UIフィールドの値から取得されます。
タイムピッカーを使用している場合は、
var maxDay = dtpAttendance.MaxDate.Day;
しかし、DateTimeオブジェクトから取得しようとしています。だからこれがあれば...
DateTime dt = DateTime.today;
月の初日と末日を取得する方法はdt
?
回答:
DateTime
構造体は、値の範囲ではなく、1つの値のみを格納します。MinValue
とMaxValue
は静的フィールドであり、DateTime
構造体のインスタンスの可能な値の範囲を保持します。これらのフィールドは静的であり、の特定のインスタンスには関連していませんDateTime
。それらはDateTime
タイプ自体に関連しています。
推奨される読み物:静的(C#リファレンス)
更新:月の範囲を取得しています:
DateTime date = ...
var firstDayOfMonth = new DateTime(date.Year, date.Month, 1);
var lastDayOfMonth = firstDayOfMonth.AddMonths(1).AddDays(-1);
lastDayofMonth
んfirstDayOfMonth.AddMonths(1).AddSeconds(-1);
か?
AddTicks(-1)
、時間の部分を気にせず、日付の部分だけを考えれば、日はうまくいきます
< firstDayOfNextMonth
代わりにしてい<= lastDayOfMonth
ます。そうすれば、粒度に関係なく常に機能します。(ティックは大丈夫だと確信していますが、未来が何をもたらすのか誰が知っています...ナノティック?)
これは、@ Sergeyと@Steffenの回答に対するより長いコメントです。過去に自分でも同様のコードを書いたことがあるので、わかりやすさも重要であることを思い出しながら、最もパフォーマンスの高いものを確認することにしました。
以下は、1,000万回の反復のテスト実行結果の例です。
2257 ms for FirstDayOfMonth_AddMethod()
2406 ms for FirstDayOfMonth_NewMethod()
6342 ms for LastDayOfMonth_AddMethod()
4037 ms for LastDayOfMonth_AddMethodWithDaysInMonth()
4160 ms for LastDayOfMonth_NewMethod()
4212 ms for LastDayOfMonth_NewMethodWithReuseOfExtMethod()
2491 ms for LastDayOfMonth_SpecialCase()
私が使用LINQPad 4をオンにコンパイラの最適化とテストを実行する(C#プログラムモードで)。以下は、明確さと利便性のために、拡張メソッドとして因数分解されたテスト済みコードです。
public static class DateTimeDayOfMonthExtensions
{
public static DateTime FirstDayOfMonth_AddMethod(this DateTime value)
{
return value.Date.AddDays(1 - value.Day);
}
public static DateTime FirstDayOfMonth_NewMethod(this DateTime value)
{
return new DateTime(value.Year, value.Month, 1);
}
public static DateTime LastDayOfMonth_AddMethod(this DateTime value)
{
return value.FirstDayOfMonth_AddMethod().AddMonths(1).AddDays(-1);
}
public static DateTime LastDayOfMonth_AddMethodWithDaysInMonth(this DateTime value)
{
return value.Date.AddDays(DateTime.DaysInMonth(value.Year, value.Month) - value.Day);
}
public static DateTime LastDayOfMonth_SpecialCase(this DateTime value)
{
return value.AddDays(DateTime.DaysInMonth(value.Year, value.Month) - 1);
}
public static int DaysInMonth(this DateTime value)
{
return DateTime.DaysInMonth(value.Year, value.Month);
}
public static DateTime LastDayOfMonth_NewMethod(this DateTime value)
{
return new DateTime(value.Year, value.Month, DateTime.DaysInMonth(value.Year, value.Month));
}
public static DateTime LastDayOfMonth_NewMethodWithReuseOfExtMethod(this DateTime value)
{
return new DateTime(value.Year, value.Month, value.DaysInMonth());
}
}
void Main()
{
Random rnd = new Random();
DateTime[] sampleData = new DateTime[10000000];
for(int i = 0; i < sampleData.Length; i++) {
sampleData[i] = new DateTime(1970, 1, 1).AddDays(rnd.Next(0, 365 * 50));
}
GC.Collect();
System.Diagnostics.Stopwatch sw = System.Diagnostics.Stopwatch.StartNew();
for(int i = 0; i < sampleData.Length; i++) {
DateTime test = sampleData[i].FirstDayOfMonth_AddMethod();
}
string.Format("{0} ms for FirstDayOfMonth_AddMethod()", sw.ElapsedMilliseconds).Dump();
GC.Collect();
sw.Restart();
for(int i = 0; i < sampleData.Length; i++) {
DateTime test = sampleData[i].FirstDayOfMonth_NewMethod();
}
string.Format("{0} ms for FirstDayOfMonth_NewMethod()", sw.ElapsedMilliseconds).Dump();
GC.Collect();
sw.Restart();
for(int i = 0; i < sampleData.Length; i++) {
DateTime test = sampleData[i].LastDayOfMonth_AddMethod();
}
string.Format("{0} ms for LastDayOfMonth_AddMethod()", sw.ElapsedMilliseconds).Dump();
GC.Collect();
sw.Restart();
for(int i = 0; i < sampleData.Length; i++) {
DateTime test = sampleData[i].LastDayOfMonth_AddMethodWithDaysInMonth();
}
string.Format("{0} ms for LastDayOfMonth_AddMethodWithDaysInMonth()", sw.ElapsedMilliseconds).Dump();
GC.Collect();
sw.Restart();
for(int i = 0; i < sampleData.Length; i++) {
DateTime test = sampleData[i].LastDayOfMonth_NewMethod();
}
string.Format("{0} ms for LastDayOfMonth_NewMethod()", sw.ElapsedMilliseconds).Dump();
GC.Collect();
sw.Restart();
for(int i = 0; i < sampleData.Length; i++) {
DateTime test = sampleData[i].LastDayOfMonth_NewMethodWithReuseOfExtMethod();
}
string.Format("{0} ms for LastDayOfMonth_NewMethodWithReuseOfExtMethod()", sw.ElapsedMilliseconds).Dump();
for(int i = 0; i < sampleData.Length; i++) {
sampleData[i] = sampleData[i].FirstDayOfMonth_AddMethod();
}
GC.Collect();
sw.Restart();
for(int i = 0; i < sampleData.Length; i++) {
DateTime test = sampleData[i].LastDayOfMonth_SpecialCase();
}
string.Format("{0} ms for LastDayOfMonth_SpecialCase()", sw.ElapsedMilliseconds).Dump();
}
これらの結果のいくつかに驚きました。
それほど多くはありませんが、テストのほとんどの実行FirstDayOfMonth_AddMethod
よりもわずかに高速FirstDayOfMonth_NewMethod
でした。ただし、後者の方がやや明確な意図があるため、私はそれを好みます。
LastDayOfMonth_AddMethod
明確な敗者だったLastDayOfMonth_AddMethodWithDaysInMonth
、LastDayOfMonth_NewMethod
とLastDayOfMonth_NewMethodWithReuseOfExtMethod
。最速の3つの間には何もないので、個人の好みによって決まります。LastDayOfMonth_NewMethodWithReuseOfExtMethod
別の便利な拡張メソッドを再利用して、の明確さを選択します。私見はその意図がより明確であり、私は小さなパフォーマンスコストを受け入れても構わないと思っています。
LastDayOfMonth_SpecialCase
は、その日付をすでに計算している可能性がある特別な場合に月の最初の日を提供し、addメソッドを DateTime.DaysInMonth
、結果を取得するために。予想通り、これは他のバージョンよりも高速ですが、スピードを切実に必要としない限り、この特殊なケースを武器に持つ意味はわかりません。
以下は、私の選択を含む拡張メソッドクラスであり、@ Steffenとの一般的な合意は私が信じています。
public static class DateTimeDayOfMonthExtensions
{
public static DateTime FirstDayOfMonth(this DateTime value)
{
return new DateTime(value.Year, value.Month, 1);
}
public static int DaysInMonth(this DateTime value)
{
return DateTime.DaysInMonth(value.Year, value.Month);
}
public static DateTime LastDayOfMonth(this DateTime value)
{
return new DateTime(value.Year, value.Month, value.DaysInMonth());
}
}
ここまで来たら、ありがとうございました!それは楽しかったです:¬)。これらのアルゴリズムについて他に提案がある場合はコメントしてください。
LastDayOfMonth_AddMethod_SpecialCase
またはそのようなもの)。パラメータとして月の最初の日を期待して、私は何が何をすべきかを最も速くすべきだと思いますLastDayOfMonth_AddMethod
!:それはのようにシンプルになりそうreturn value.AddMonths(1).AddDays(-1);
.Net APIを使用して月の範囲を取得する(別の方法):
DateTime date = ...
var firstDayOfMonth = new DateTime(date.Year, date.Month, 1);
var lastDayOfMonth = new DateTime(date.Year, date.Month, DateTime.DaysInMonth(date.Year, date.Month));
「Last day of month
」は実際には「First day of *next* month, minus 1
」です。だから私はこれを使っています、「DaysInMonth」メソッドは必要ありません:
public static DateTime FirstDayOfMonth(this DateTime value)
{
return new DateTime(value.Year, value.Month, 1);
}
public static DateTime LastDayOfMonth(this DateTime value)
{
return value.FirstDayOfMonth()
.AddMonths(1)
.AddMinutes(-1);
}
注:ここではAddMinutes(-1)
なく、私が使用する理由AddDays(-1)
は、通常、特定の期間のレポートにこれらの日付関数が必要であり、ある期間のレポートを作成する場合、「終了日」は実際にはOct 31 2015 23:59:59
レポートが正しく機能するようにする必要があります-月の最終日からのすべてのデータを含みます。
つまり、実際には「月の最後の瞬間」がここに表示されます。最終日ではありません。
了解しました。今から黙ります。
DateTime dCalcDate = DateTime.Now;
dtpFromEffDate.Value = new DateTime(dCalcDate.Year, dCalcDate.Month, 1);
dptToEffDate.Value = new DateTime(dCalcDate.Year, dCalcDate.Month, DateTime.DaysInMonth(dCalcDate.Year, dCalcDate.Month));
description
コードの説明に役立つを追加することを検討してください。ありがとう
日付だけ気にするなら
var firstDay = new DateTime(date.Year, date.Month, 1, 0, 0, 0, date.Kind);
var lastDay = new DateTime(date.Year, date.Month, 1, 0, 0, 0, date.Kind).AddMonths(1).AddDays(-1);
時間を守りたいなら
var firstDay = new DateTime(date.Year, date.Month, 1, date.Hour, date.Minute, date.Second, date.Kind);
var lastDay = new DateTime(date.Year, date.Month, 1, date.Hour, date.Minute, date.Second, date.Kind).AddMonths(1).AddDays(-1);
ここで受け入れられた回答は、DateTimeインスタンスの種類を考慮していません。たとえば、元のDateTimeインスタンスがUTCの種類であった場合、新しいDateTimeインスタンスを作成することで、不明な種類のインスタンスを作成し、サーバーの設定に基づいてローカル時間として処理されます。したがって、月の最初と最後の日付を取得するより適切な方法は次のとおりです。
var now = DateTime.UtcNow;
var first = now.Date.AddDays(-(now.Date.Day - 1));
var last = first.AddMonths(1).AddTicks(-1);
これにより、元の種類のDateTimeインスタンスが保持されます。
これを試してみてください。基本的には、経過した日数を計算し、その日数DateTime.Now
から1を引いて、新しい値を使用して現在の月の最初の日を見つけます。そこからそれを使用し、前の月の最初を取得するためにDateTime
使用.AddMonths(-1)
します。
先月の最終日を取得することは基本的に同じことですが、月の日数に1を加え、その値をから差し引いて、前月DateTime.Now.AddDays
の最終日を与えます。
int NumberofDays = DateTime.Now.Day;
int FirstDay = NumberofDays - 1;
int LastDay = NumberofDays + 1;
DateTime FirstofThisMonth = DateTime.Now.AddDays(-FirstDay);
DateTime LastDayOfLastMonth = DateTime.Now.AddDays(-LastDay);
DateTime CheckLastMonth = FirstofThisMonth.AddMonths(-1);
ペルシャ文化のために
PersianCalendar pc = new PersianCalendar();
var today = pc.GetDayOfMonth(DateTime.Now);
var firstDayOfMonth = pc.GetDayOfMonth(DateTime.Now.AddDays(-(today-1)));
var lastDayOfMonth = pc.GetDayOfMonth(DateTime.Now.AddMonths(1).AddDays(-today));
Console.WriteLine("First day "+ firstDayOfMonth);
Console.WriteLine("Last day " + lastDayOfMonth);
それを行う簡単な方法
Begin = new DateTime(DateTime.Now.Year, DateTime.Now.Month,1).ToShortDateString();
End = new DataFim.Text = new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.DaysInMonth(DateTime.Now.Year, DateTime.Now.Month)).ToShortDateString();
_Date
変数には単一の値が格納されています。その値から何を取得しようとしていますか?