在c#中安全地向DateTime添加/减去日期

本文关键字:添加 日期 DateTime 安全 | 更新日期: 2023-09-27 18:01:38

我有一个函数,它接受两个DateTime参数,我必须为这些日期添加单独的偏移量。我知道DateTime有一个AddDays函数来向日期添加天数,如果DateTime小于MinValue或大于MaxValue,它会抛出异常。

现在我想做一个安全的检查,添加/减去DateTime的以下天数是否会导致流量过大/不足。

safeStartDate = (startDate == DateTime.MinValue || startDate == DateTime.MaxValue) ? startDate : startDate.AddDays(startDateOffset);
safeEndDate = (endDate == DateTime.MaxValue || endDate == DateTime.MinValue) ? endDate : endDate.AddDays(enDateOffset);

通过这样做,我使它一个级别的异常自由,但日期可以是DateTime.Max - 1,而试图添加偏移,它抛出一个异常。我正在寻找一种更好的方法,即是否最终值在不进行实际计算的情况下流过/流过,以防止异常。

在c#中安全地向DateTime添加/减去日期

如果不经常调用catch,您可以:

try
{
    safeDate = dt.AddDays(days);
}
catch (ArgumentOutOfRangeException)
{
    safeDate = date;
}

或者

var maxDays = (DateTime.MaxValue - dt).TotalDays;
safeDate = (days <= maxDays) ? dt.AddDays(days) : dt;

如果是负数:

var maxDays = (DateTime.MaxValue - dt).TotalDays;
var minDays = (DateTime.MinValue - dt).TotalDays;
return (minDays <= days && days <= maxDays) ? dt.AddDays(days) : dt;

或者直接使用罗林回答中的方法:CanAddDays(dt, days) ? dt.AddDays(days) : dt

如果你不捕获,try/catch版本大约快25%,如果你捕获,大约慢1000倍。因此,如果您希望在每5000次使用中捕获超过1次,那么请使用第二个版本。

您可以使用以下命令来检查是否可以为给定的DateTime添加给定的天数而不会导致溢出:

bool CanAddDays(DateTime dt, int days)
{
    double maxDaysToAdd = (DateTime.MaxValue - dt).TotalDays;
    double minDaysToAdd = (DateTime.MinValue - dt).TotalDays;
    return days <= maxDaysToAdd && days >= minDaysToAdd;
}

您可以考虑以下方法:

private static DateTime AddDays(DateTime dateTime, int days)
{
    var daysTimeSpanTicks = (new TimeSpan(days, 0, 0, 0)).Ticks;
    return (days >= 0) ?
        (DateTime.MaxValue.Ticks < dateTime.Ticks + daysTimeSpanTicks) ? dateTime : dateTime.AddDays(days) :
        (dateTime.Ticks + daysTimeSpanTicks < 0) ? dateTime : dateTime.AddDays(days);
}

示例用法如下:

DateTime date = DateTime.MinValue;
DateTime safe = AddDays(date, -100);

我猜你正在寻找这样的东西

DateTime Now = DateTime.Now;
DateTime Max = DateTime.MaxValue;
Max.Subtract(Now);
int DaysToAdd = 1000;//or something else
if (Max.Day > DaysToAdd) Now.AddDays(DaysToAdd);//add