C # DateTime: Какую «дату» использовать, когда я использую только «время»? - PullRequest
30 голосов
/ 16 декабря 2008

Я использую DateTime в C # для отображения времени. Какую часть даты каждый использует при построении времени?

например. следующее недопустимо, потому что нет нулевого месяца или нулевого дня:

// 4:37:58 PM
DateTime time = new DateTime(0, 0, 0, 16, 47, 58);

Использую ли я нулевую дату COM?

// 4:37:58 PM
DateTime time = new DateTime(1899, 12, 30, 16, 47, 58);

Или, возможно, SQL Server?

//4:37:58 PM
DateTime time = new DateTime(1900, 1, 1, 16, 47, 58);

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

DateTime duration = time2 - time1;

Ответ

Мне кажется, мне нравится MinValue

 DateTime time = DateTime.MinValue.Date.Add(new TimeSpan(16, 47, 58));

Примечание: Я не могу использовать TimeSpan, потому что это не хранит время суток. И причина, по которой я это знаю, заключается в том, что нет способа отобразить его содержимое как время.

То есть TimeSpan записывает промежуток времени , а не время суток , например ::

TimeSpan t = new TimeSpan(16, 47, 58);
t.ToString();

возвращает промежуток времени в формате часов : минут : секунд , например ::

16:47:58

а не время:

4:47:58 PM    (United States)
04:47:58 nm   (South Africa)
4:47:58.MD    (Albania)
16:47:58      (Algeria)
04:47:58 م    (Bahrain)
PM 4:47:58    (Singapore)
下午 04:47:58  (Taiwan)
04:47:58 PM   (Belize)
4:47:58 p.m.  (New Zealand)
4:47:58 μμ    (Greece)
16.47.58      (Italy)
오후 4:47:58   (Korea)
04:47:58 ب.ظ  (Iran)
ਸ਼ਾਮ 04:47:58   (India)
04:47:58 p.m. (Argentina)
etc

Другими словами, есть разница между временным интервалом и временем. А также осознайте, что TimeSpan не предоставляет механизма для преобразования промежутка времени в время суток - и для этого есть причина.

Ответы [ 12 ]

25 голосов
/ 16 декабря 2008

а как насчет DateTime.MinValue?

8 голосов
/ 19 мая 2009

Давайте поможем ребятам, которые хотят структуру времени:

/// <summary>
/// Time structure
/// </summary>
public struct Time : IComparable
{
    private int minuteOfDay;
    public static Time Midnight = "0:00";
    private static int MIN_OF_DAY = 60 * 24;

    public Time(int minuteOfDay)
    {
        if (minuteOfDay >= (60 * 24) || minuteOfDay < 0)
            throw new ArgumentException("Must be in the range 0-1439", "minuteOfDay");
        this.minuteOfDay = minuteOfDay;
    }

    public Time(int hour, int minutes)
    {
        if (hour < 0 || hour > 23)
            throw new ArgumentException("Must be in the range 0-23", "hour");
        if (minutes < 0 || minutes > 59)
            throw new ArgumentException("Must be in the range 0-59", "minutes");

        minuteOfDay = (hour * 60) + minutes;
    }

    #region Operators
    public static implicit operator Time(string s)
    {
        var parts = s.Split(':');
        if (parts.Length != 2)
            throw new ArgumentException("Time must be specified on the form tt:mm");
        return new Time(int.Parse(parts[0]), int.Parse(parts[1]));
    }


    public static bool operator >(Time t1, Time t2)
    {
        return t1.MinuteOfDay > t2.MinuteOfDay;
    }
    public static bool operator <(Time t1, Time t2)
    {
        return t1.MinuteOfDay < t2.MinuteOfDay;
    }
    public static bool operator >=(Time t1, Time t2)
    {
        return t1.MinuteOfDay >= t2.MinuteOfDay;
    }
    public static bool operator <=(Time t1, Time t2)
    {
        return t1.MinuteOfDay <= t2.MinuteOfDay;
    }
    public static bool operator ==(Time t1, Time t2)
    {
        return t1.GetHashCode() == t2.GetHashCode();
    }
    public static bool operator !=(Time t1, Time t2)
    {
        return t1.GetHashCode() != t2.GetHashCode();
    }

    /// Time
    /// Minutes that remain to
    /// Time conferred minutes
    public static Time operator +(Time t, int min)
    {
        if (t.minuteOfDay + min < (24 * 60))
        {
            t.minuteOfDay += min;
            return t;
        }
        else
        {
            t.minuteOfDay = (t.minuteOfDay + min) % MIN_OF_DAY;
            return t;
        }
    }

    public static Time operator -(Time t, int min)
    {
        if (t.minuteOfDay - min > -1)
        {
            t.minuteOfDay -= min;
            return t;
        }
        else
        {
            t.minuteOfDay = MIN_OF_DAY + (t.minuteOfDay - min);
            return t;
        }
    }

    public static TimeSpan operator -(Time t1, Time t2)
    {
        return TimeSpan.FromMinutes(Time.Span(t2, t1));
    }
    #endregion


    public int Hour
    {
        get
        {
            return (int)(minuteOfDay / 60);
        }
    }
    public int Minutes
    {
        get
        {
            return minuteOfDay % 60;
        }
    }


    public int MinuteOfDay
    {
        get { return minuteOfDay; }
    }

    public Time AddHours(int hours)
    {
        return this + (hours * 60);
    }

    public int CompareTo(Time other)
    {
        return this.minuteOfDay.CompareTo(other.minuteOfDay);
    }

    #region Overrides
    public override int GetHashCode()
    {
        return minuteOfDay.GetHashCode();
    }

    public override string ToString()
    {
        return string.Format("{0}:{1:00}", Hour, Minutes);
    }
    #endregion

    /// 
    /// Safe enumerering - whatever interval applied max days 
    /// 
    /// Start time
    /// Spring in minutes
    /// 
    public static IEnumerable Range(Time start, int step)
    {
        return Range(start, start, step);
    }

    /// 
    /// Safe enumeration - whatever interval applied max days
    /// 
    public static IEnumerable Range(Time start, Time stop, int step)
    {
        int offset = start.MinuteOfDay;
        for (var i = 0; i < Time.Span(start, stop); i += step)
        {
            yield return Time.Midnight + (i + offset);
        }
    }

    /// 
    /// Calculates the number of minutes between t1 and t2
    /// 
    public static int Span(Time t1, Time t2)
    {
        if (t1 < t2) // same day
            return t2.MinuteOfDay - t1.MinuteOfDay;
        else // over midnight
            return MIN_OF_DAY - t1.MinuteOfDay + t2.MinuteOfDay;
    }
}
6 голосов
/ 17 декабря 2008

Лично я бы создал пользовательский Time struct, который содержит экземпляр DateTime, который имеет схожие свойства, конструкторы и т. Д., Но не содержит дней, месяцев и т. Д. Просто сделайте так, чтобы все ваши публичные средства доступа проходили через содержащийся экземпляр. Тогда вы можете просто указать эпоху как поле private static readonly DateTime, и не имеет значения, какое значение вы выберете, поскольку все это аккуратно содержится в вашей пользовательской структуре. В остальном свой код можно просто написать:

var time = new Time(16, 47, 58);
6 голосов
/ 16 декабря 2008

TimeSpan, несомненно, может хранить время суток - вам просто нужно воспринимать значение как количество времени, прошедшего с полуночи, в основном так же, как мы читаем часы.

4 голосов
/ 16 декабря 2008

Учитывая, что DateTime.TimeOfDay возвращает TimeSpan, я бы использовал это.

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

3 голосов
/ 17 декабря 2008

Чтобы отобразить TimeSpan, отформатированный с учетом местной культуры, просто добавьте его к дате, например DateTime.Today. Примерно так:

(DateTime.Today + timeSpan) .ToString ();

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

3 голосов
/ 16 декабря 2008

Как насчет DateTime.Now.TimeOfDay, и использовать TimeSpan?

Re "потому что это не хранит время суток." - ну, это так, если вы думаете о TimeSpan как о времени с полуночи.

«Продолжительность», например, крики TimeSpan.

1 голос
/ 17 декабря 2008

Могу ли я предположить, что в некоторых случаях может подойти пользовательская структура? Может иметь значение поддержки Int32 (в день 86 миллионов миллисекунд; это будет соответствовать Int32).

Могут быть свойства только для получения:

Часы минут секунд Миллисекунды

Вы также можете перегрузить операторы, такие как +, - и так далее. Реализуйте IEquatable, IComparable и все, что может быть. Перегрузка равна, ==. Перегрузка и переопределение ToString.

Вы также можете предоставить больше методов для конструирования из DateTime или добавления к дате и т. Д.

1 голос
/ 17 декабря 2008

Вы можете просто создать новый DateTime со строковым литералом.

Строковый литерал для времени:

DateTime t = new DateTime("01:00:30");

Строковый литерал для даты:

DateTime t = new DateTime("01/05/2008"); // english format
DateTime t = new DateTime("05.01.2008"); // german format

Для DateTime со значениями даты и времени:

DateTime t = new DateTime("01/05/2008T01:00:30");

В большинстве случаев при создании DateTime я устанавливаю его в DateTime.Now, если он на самом деле не установлен ни на что другое. Если вы создаете экземпляр DateTime вручную, вам следует остерегаться правильной установки DateTimeKind, в противном случае это может привести к неожиданностям.

1 голос
/ 16 декабря 2008

Я рекомендую DateTime.MinValue

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...