/******************************************************************************* * Copyright © 2017-2020 Znyc.Recruitment.Admin.Framework 版权所有 * Author: Znyc * Description: Znyc快速开发平台 * Website:http://www.Znyc.Recruitment.Admin.com *********************************************************************************/ using System; namespace Znyc.Recruitment.Admin.Commons.Extensions { /// /// 日期时间扩展 /// public static class DateTimeExtensions { // Number of 100ns ticks per time unit private const long TicksPerMillisecond = 10000; private const long TicksPerSecond = TicksPerMillisecond * 1000; private const long TicksPerMinute = TicksPerSecond * 60; private const long TicksPerHour = TicksPerMinute * 60; private const long TicksPerDay = TicksPerHour * 24; // Number of milliseconds per time unit private const int MillisPerSecond = 1000; private const int MillisPerMinute = MillisPerSecond * 60; private const int MillisPerHour = MillisPerMinute * 60; private const int MillisPerDay = MillisPerHour * 24; // Number of days in a non-leap year private const int DaysPerYear = 365; // Number of days in 4 years private const int DaysPer4Years = DaysPerYear * 4 + 1; // 1461 // Number of days in 100 years private const int DaysPer100Years = DaysPer4Years * 25 - 1; // 36524 // Number of days in 400 years private const int DaysPer400Years = DaysPer100Years * 4 + 1; // 146097 // Number of days from 1/1/0001 to 12/31/1600 private const int DaysTo1601 = DaysPer400Years * 4; // 584388 // Number of days from 1/1/0001 to 12/30/1899 private const int DaysTo1899 = DaysPer400Years * 4 + DaysPer100Years * 3 - 367; // Number of days from 1/1/0001 to 12/31/1969 internal const int DaysTo1970 = DaysPer400Years * 4 + DaysPer100Years * 3 + DaysPer4Years * 17 + DaysPerYear; // 719,162 // Number of days from 1/1/0001 to 12/31/9999 private const int DaysTo10000 = DaysPer400Years * 25 - 366; // 3652059 internal const long MinTicks = 0; internal const long MaxTicks = DaysTo10000 * TicksPerDay - 1; private const long MaxMillis = (long)DaysTo10000 * MillisPerDay; private const long DoubleDateOffset = DaysTo1899 * TicksPerDay; // All OA dates must be greater than (not >=) OADateMinAsDouble private const double OADateMinAsDouble = -657435.0; // All OA dates must be less than (not <=) OADateMaxAsDouble private const double OADateMaxAsDouble = 2958466.0; /// /// public static readonly DateTime BeginOfEpoch = new(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc); /// /// Converts a nullable date/time value to UTC. /// /// The nullable date/time /// The nullable date/time in UTC public static DateTime? ToUniversalTime(this DateTime? dateTime) { return dateTime.HasValue ? dateTime.Value.ToUniversalTime() : null; } /// /// Converts a nullable UTC date/time value to local time. /// /// The nullable UTC date/time /// The nullable UTC date/time as local time public static DateTime? ToLocalTime(this DateTime? dateTime) { return dateTime.HasValue ? dateTime.Value.ToLocalTime() : null; } /// /// Returns a date that is rounded to the next even hour above the given /// date. ///

/// For example an input date with a time of 08:13:54 would result in a date /// with the time of 09:00:00. If the date's time is in the 23rd hour, the /// date's 'day' will be promoted, and the time will be set to 00:00:00. ///

///
/// /// the Date to round, if the current time will /// be used /// /// the new rounded date public static DateTime GetEvenHourDate(this DateTime? dateTime) { if (!dateTime.HasValue) { dateTime = DateTime.UtcNow; } DateTime d = dateTime.Value.AddHours(1); return new DateTime(d.Year, d.Month, d.Day, d.Hour, 0, 0); } /// /// Returns a date that is rounded to the next even minute above the given /// date. ///

/// For example an input date with a time of 08:13:54 would result in a date /// with the time of 08:14:00. If the date's time is in the 59th minute, /// then the hour (and possibly the day) will be promoted. ///

///
/// The Date to round, if the current time will be used /// The new rounded date public static DateTime GetEvenMinuteDate(this DateTime? dateTime) { if (!dateTime.HasValue) { dateTime = DateTime.UtcNow; } DateTime d = dateTime.Value; d = d.AddMinutes(1); return new DateTime(d.Year, d.Month, d.Day, d.Hour, d.Minute, 0); } /// /// Returns a date that is rounded to the previous even minute below the /// given date. ///

/// For example an input date with a time of 08:13:54 would result in a date /// with the time of 08:13:00. ///

///
/// /// the Date to round, if the current time will /// be used /// /// the new rounded date public static DateTime GetEvenMinuteDateBefore(this DateTime? dateTime) { if (!dateTime.HasValue) { dateTime = DateTime.UtcNow; } DateTime d = dateTime.Value; return new DateTime(d.Year, d.Month, d.Day, d.Hour, d.Minute, 0); } /// /// /// /// public static long ToJavaScriptTicks(this DateTime dateTime) { DateTimeOffset utcDateTime = dateTime.ToUniversalTime(); long javaScriptTicks = (utcDateTime.Ticks - BeginOfEpoch.Ticks) / 10000; return javaScriptTicks; } /// /// Get the first day of the month for /// any full date submitted /// /// /// public static DateTime GetFirstDayOfMonth(this DateTime date) { DateTime dtFrom = date; dtFrom = dtFrom.AddDays(-(dtFrom.Day - 1)); return dtFrom; } /// /// Get the last day of the month for any /// full date /// /// /// public static DateTime GetLastDayOfMonth(this DateTime date) { DateTime dtTo = date; dtTo = dtTo.AddMonths(1); dtTo = dtTo.AddDays(-dtTo.Day); return dtTo; } /// /// /// /// public static DateTime ToEndOfTheDay(this DateTime dt) { if (dt != null) { return new DateTime(dt.Year, dt.Month, dt.Day, 23, 59, 59); } return dt; } /// /// /// /// public static DateTime? ToEndOfTheDay(this DateTime? dt) { return dt.HasValue ? dt.Value.ToEndOfTheDay() : dt; } /// Epoch time. Number of seconds since midnight (UTC) on 1st January 1970. public static long ToUnixTime(this DateTime date) { return Convert.ToInt64((date.ToUniversalTime() - BeginOfEpoch).TotalSeconds); } /// UTC date based on number of seconds since midnight (UTC) on 1st January 1970. public static DateTime FromUnixTime(this long unixTime) { return BeginOfEpoch.AddSeconds(unixTime); } // Creates a DateTime from an OLE Automation Date. // public static DateTime FromOADate(this double d) { return new(DoubleDateToTicks(d), DateTimeKind.Unspecified); } // Converts an OLE Date to a tick count. // This function is duplicated in COMDateTime.cpp internal static long DoubleDateToTicks(double value) { // The check done this way will take care of NaN if (!(value < OADateMaxAsDouble) || !(value > OADateMinAsDouble)) { throw new ArgumentException("Arg_OleAutDateInvalid"); } // Conversion to long will not cause an overflow here, as at this point the "value" is in between OADateMinAsDouble and OADateMaxAsDouble long millis = (long)(value * MillisPerDay + (value >= 0 ? 0.5 : -0.5)); // The interesting thing here is when you have a value like 12.5 it all positive 12 days and 12 hours from 01/01/1899 // However if you a value of -12.25 it is minus 12 days but still positive 6 hours, almost as though you meant -11.75 all negative // This line below fixes up the millis in the negative case if (millis < 0) { millis -= millis % MillisPerDay * 2; } millis += DoubleDateOffset / TicksPerMillisecond; if (millis < 0 || millis >= MaxMillis) { throw new ArgumentException("Arg_OleAutDateScale"); } return millis * TicksPerMillisecond; } #region 时间对象与时间戳相互转换 /// /// 将 DateTime 转换为 Unix 时间戳 /// /// 需要转换的时间 /// Unix 时间戳 public static long DateTimeToUnixTime(this DateTime dateTime) { return (long)(dateTime - TimeZoneInfo.ConvertTime(new DateTime(1970, 1, 1, 0, 0, 0), TimeZoneInfo.Local)) .TotalSeconds; } /// /// 将 Unix 时间戳转换为 DateTime /// /// Unix 时间戳 /// 需要转换的时间 public static DateTime UnixTimeToDateTime(this long timestamp) { if (timestamp < 0) { throw new ArgumentOutOfRangeException("timestamp is out of range"); } return TimeZoneInfo.ConvertTime(new DateTime(1970, 1, 1, 0, 0, 0), TimeZoneInfo.Local) .AddSeconds(timestamp); } #endregion 时间对象与时间戳相互转换 } }