Как урезать миллисекунды с .NET DateTime

Я пытаюсь сравнить метку времени с входящим запросом на хранимое значение базы данных. SQL Server, конечно же, сохраняет некоторую точность миллисекунд в то время, а при чтении в .NET DateTime он включает эти миллисекунды. Однако входящий запрос в систему не дает такой точности, поэтому мне нужно просто сбросить миллисекунды.

Я чувствую, что мне не хватает чего-то очевидного, но я не нашел элегантный способ сделать это (С#).

Ответ 1

Следующее будет работать для DateTime, который имеет дробные миллисекунды, а также сохраняет свойство Kind (Local, Utc или Undefined).

DateTime dateTime = ... anything ...
dateTime = new DateTime(
    dateTime.Ticks - (dateTime.Ticks % TimeSpan.TicksPerSecond), 
    dateTime.Kind
    );

или эквивалент и короче:

dateTime = dateTime.AddTicks( - (dateTime.Ticks % TimeSpan.TicksPerSecond));

Это можно обобщить в метод расширения:

public static DateTime Truncate(this DateTime dateTime, TimeSpan timeSpan)
{
    if (timeSpan == TimeSpan.Zero) return dateTime; // Or could throw an ArgumentException
    if (dateTime == DateTime.MinValue || dateTime == DateTime.MaxValue) return dateTime; // do not modify "guard" values
    return dateTime.AddTicks(-(dateTime.Ticks % timeSpan.Ticks));
}

который используется следующим образом:

dateTime = dateTime.Truncate(TimeSpan.FromMilliseconds(1)); // Truncate to whole ms
dateTime = dateTime.Truncate(TimeSpan.FromSeconds(1)); // Truncate to whole second
dateTime = dateTime.Truncate(TimeSpan.FromMinutes(1)); // Truncate to whole minute
...

Ответ 2

var date = DateTime.Now;

date = new DateTime(date.Year, date.Month, date.Day, date.Hour, date.Minute, date.Second, date.Kind);

Ответ 3

Вот метод расширения, основанный на предыдущем ответе, который позволит вам усекать любое разрешение...

Применение:

DateTime myDateSansMilliseconds = myDate.Truncate(TimeSpan.TicksPerSecond);
DateTime myDateSansSeconds = myDate.Truncate(TimeSpan.TicksPerMinute)

Класс:

public static class DateTimeUtils
{
    /// <summary>
    /// <para>Truncates a DateTime to a specified resolution.</para>
    /// <para>A convenient source for resolution is TimeSpan.TicksPerXXXX constants.</para>
    /// </summary>
    /// <param name="date">The DateTime object to truncate</param>
    /// <param name="resolution">e.g. to round to nearest second, TimeSpan.TicksPerSecond</param>
    /// <returns>Truncated DateTime</returns>
    public static DateTime Truncate(this DateTime date, long resolution)
    {
        return new DateTime(date.Ticks - (date.Ticks % resolution), date.Kind);
    }
}

Ответ 4

DateTime d = DateTime.Now;
d = d.AddMilliseconds(-d.Millisecond);

Ответ 5

Вместо того, чтобы отбрасывать миллисекунды, тогда сравнение, почему бы не сравнить разницу?

DateTime x; DateTime y;
bool areEqual = (x-y).TotalSeconds == 0;

или

TimeSpan precision = TimeSpan.FromSeconds(1);
bool areEqual = (x-y).Duration() < precision;

Ответ 6

Иногда вы хотите усечь что-то на основе календаря, например, год или месяц. Здесь используется метод расширения, позволяющий выбрать любое разрешение.

public enum DateTimeResolution
{
    Year, Month, Day, Hour, Minute, Second, Millisecond, Tick
}

public static DateTime Truncate(this DateTime self, DateTimeResolution resolution = DateTimeResolution.Second)
{
    switch (resolution)
    {
        case DateTimeResolution.Year:
            return new DateTime(self.Year, 1, 1, 0, 0, 0, 0, self.Kind);
        case DateTimeResolution.Month:
            return new DateTime(self.Year, self.Month, 1, 0, 0, 0, self.Kind);
        case DateTimeResolution.Day:
            return new DateTime(self.Year, self.Month, self.Day, 0, 0, 0, self.Kind);
        case DateTimeResolution.Hour:
            return self.AddTicks(-(self.Ticks % TimeSpan.TicksPerHour));
        case DateTimeResolution.Minute:
            return self.AddTicks(-(self.Ticks % TimeSpan.TicksPerMinute));
        case DateTimeResolution.Second:
            return self.AddTicks(-(self.Ticks % TimeSpan.TicksPerSecond));
        case DateTimeResolution.Millisecond:
            return self.AddTicks(-(self.Ticks % TimeSpan.TicksPerMillisecond));
        case DateTimeResolution.Tick:
            return self.AddTicks(0);
        default:
            throw new ArgumentException("unrecognized resolution", "resolution");
    }
}

Ответ 7

Менее очевидна, но более чем в 2 раза быстрее:

// 10000000 runs

DateTime d = DateTime.Now;

// 484,375ms
d = new DateTime((d.Ticks / TimeSpan.TicksPerSecond) * TimeSpan.TicksPerSecond);

// 1296,875ms
d = d.AddMilliseconds(-d.Millisecond);

Ответ 8

Простой...

//Remove milliseconds
DateTime date = DateTime.Now;
date = DateTime.ParseExact(date.ToString("yyyy-MM-dd HH:mm:ss"), "yyyy-MM-dd HH:mm:ss", null);

И еще...

//Remove seconds
DateTime date = DateTime.Now;
date = DateTime.ParseExact(date.ToString("yyyy-MM-dd HH:mm"), "yyyy-MM-dd HH:mm", null);

//Remove minutes
DateTime date = DateTime.Now;
date = DateTime.ParseExact(date.ToString("yyyy-MM-dd HH"), "yyyy-MM-dd HH", null);

//and go on...

Ответ 9

Чтобы округлить до второго:

dateTime.AddTicks(-dateTime.Ticks % TimeSpan.TicksPerSecond)

Замените TicksPerMinute округлением до минуты.


Если ваш код чувствителен к производительности, будьте осторожны в отношении

new DateTime(date.Year, date.Month, date.Day, date.Hour, date.Minute, date.Second)

Мое приложение тратило 12% процессорного времени в System.DateTime.GetDatePart.

Ответ 10

Относительно ответа диадиста. Это сработало для меня, за исключением того, что мне пришлось использовать "Пол" для удаления дробной части деления до умножения. Таким образом,

d = new DateTime((d.Ticks / TimeSpan.TicksPerSecond) * TimeSpan.TicksPerSecond);

становится

d = new DateTime(Math.Floor(d.Ticks / TimeSpan.TicksPerSecond) * TimeSpan.TicksPerSecond);

Я бы ожидал, что деление двух длинных значений приведет к длинному, таким образом удалив десятичную часть, но она разрешит ее как двойную, оставляя то же самое значение после умножения.

Eppsy

Ответ 11

2 Методы расширения для упомянутых выше решений

    public static bool LiesAfterIgnoringMilliseconds(this DateTime theDate, DateTime compareDate, DateTimeKind kind)
    {
        DateTime thisDate = new DateTime(theDate.Year, theDate.Month, theDate.Day, theDate.Hour, theDate.Minute, theDate.Second, kind);
        compareDate = new DateTime(compareDate.Year, compareDate.Month, compareDate.Day, compareDate.Hour, compareDate.Minute, compareDate.Second, kind);

        return thisDate > compareDate;
    }


    public static bool LiesAfterOrEqualsIgnoringMilliseconds(this DateTime theDate, DateTime compareDate, DateTimeKind kind)
    {
        DateTime thisDate = new DateTime(theDate.Year, theDate.Month, theDate.Day, theDate.Hour, theDate.Minute, theDate.Second, kind);
        compareDate = new DateTime(compareDate.Year, compareDate.Month, compareDate.Day, compareDate.Hour, compareDate.Minute, compareDate.Second, kind);

        return thisDate >= compareDate;
    }

использование:

bool liesAfter = myObject.DateProperty.LiesAfterOrEqualsIgnoringMilliseconds(startDateTime, DateTimeKind.Utc);

Ответ 12

DateID.Text = DateTime.Today.ToShortDateString();

Use ToShortDateString() //Date 2-02-2016
Use ToShortDateString() // Time 

И при использовании

ToLongDateString() // its show 19 February 2016.

: P

Ответ 13

Новый метод

String Date = DateTime.Today.ToString("dd-MMM-yyyy"); 

//определить параметр прохода строки dd-mmm-yyyy return 24-feb-2016

Или показано в текстовом поле

txtDate.Text = DateTime.Today.ToString("dd-MMM-yyyy");

//нажимаем на PageonLoad

Ответ 14

В моем случае я стремился сохранить TimeSpan из инструмента datetimePicker без сохранения секунд и миллисекунд, и вот решение.

Сначала преобразуйте datetimePicker.value в нужный вам формат, который у меня равен "ЧЧ: мм", затем преобразуйте его обратно в TimeSpan.

var datetime = datetimepicker1.Value.ToString("HH:mm");
TimeSpan timeSpan = Convert.ToDateTime(datetime).TimeOfDay;

Ответ 15

Я знаю, что ответ довольно поздно, но лучший способ избавиться от миллисекунд -

var currentDateTime = DateTime.Now.ToString("s");

Попробуйте напечатать значение переменной, оно покажет дату и время без миллисекунд.