When I use the InvariantCulture
to parse a string, with both:
var christ1 = DateTime.Parse("12/25/35", CultureInfo.InvariantCulture);
and:
var christ2 = DateTime.ParseExact("12/25/35", "MM/dd/yy", CultureInfo.InvariantCulture);
the result depends on what computer the application runs on. On one machine I get the equivalent of:
December 25, 2035
while on another machine I get a date one century earlier, that is:
December 25, 1935
What is going on?
The so-called invariant culture is not so invariant in this case. It depends on both your OS version and the current user's regional settings in the OS.
In .NET terms, the property of interest is TwoDigitYearMax
on the calendar, so inspect CultureInfo.InvariantCulture.DateTimeFormat.Calendar.TwoDigitYearMax
or (new GregorianCalendar()).TwoDigitYearMax
. Documentation of GregorianCalendar.TwoDigitYearMax
override.
On a Windows 10 version 1809 ("October 2018 Update", "Redstone 5") machine where the user has not changed the regional settings (more on that below), the property TwoDigitYearMax
is 2029
, and as a consequence you get December 25, 1935.
But on Windows 10 version 1903 ("May 2019 Update", "19H1"), the same property returns 2049
, and therefore your result is December 25, 2035.
However, the value depends not only on the Windows version. Also what the user inputs in regional settings, matters. So that is not really "invariant".
You can change the cutoff year in the Control Panel by:
In older Windows flavors, it may be (pasted from microsoft.com
):
Or you can use Windows Registry directly, go to key HKEY_CURRENT_USER\Control Panel\International\Calendars\TwoDigitYearMax
, and for every relevant "value" in that key, keep the name component unchanged and edit the data component of the value to (the usual decimal string representation of) the cutoff year you desire. If the user has never changed this setting from the Control Panel, the two innermost keys (Calendars\TwoDigitYearMax
) do not exist yet. Usual warnings about mingling with the registry apply.
Once you changed the year in the Control Panel, every .NET process that is started after you made these changes, will reflect your choice in the TwoDigitYearMax
property reached through InvarantCulture
!
So if you want consistent behavior across OS versions and user preferences, I think you cannot use InvariantCulture
. Instead, something like this could work:
var tmp = (CultureInfo)(CultureInfo.InvariantCulture.Clone());
tmp.DateTimeFormat.Calendar.TwoDigitYearMax = 2039; // any cutoff you need
// incorrect: tmp.Calendar.TwoDigitYearMax = 2039
var newInvariantCulture = CultureInfo.ReadOnly(tmp);
then use that newInvariantCulture
instance whenever you parse dates with 2-digit years (you can put it in some static readonly
field).
WARNING: Each CultureInfo x
carries two distinct Calendar
s that may have different 2-digit year max, namely:
x.DateTimeFormat.Calendar.TwoDigitYearMax
!=
x.Calendar.TwoDigitYearMax