22
votes

I was executing below statement under nodejs repl and i was getting two different result for same date

var dateStr1 = "2015/03/31";
var dateStr2 = "2015-03-31";
var date1 = new Date(dateStr1);//gives Tue Mar 31 2015 00:00:00 GMT+0530 (IST)
var date2 = new Date(dateStr2);//gives Tue Mar 31 2015 05:30:00 GMT+0530 (IST)

In the 1st one hour,min,seconds are all zeros while in the 2nd one by default hour,min is getting set to as a timezone hour,min which is 5:30

2
^^ That one should probably be closed as a duplicate of this one rather than the other way around. Both the question and answer are clearer on this one, given the documentation link & quote.Michael Berkowski
@MichaelBerkowski True, I have done so.Anonymous
This question has been asked in so many different ways before, it's hard to find an exact dup. This one is very similar and quite old. Though the format is slightly different.Matt Johnson-Pint
FYI - You might consider using moment.js to work around browser differences with date parsing.Matt Johnson-Pint
And it certainly qualifies as a "gotcha!" Documented, "correct" behavior, and ... thoroughly counter-intuitive. "Now you know."Mike Robinson

2 Answers

19
votes

It boils down to how Date.parse() handles the ISO-8601 date format.

The date time string may be in ISO 8601 format. For example, "2011-10-10" (just date) or "2011-10-10T14:48:00" (date and time) can be passed and parsed. The UTC time zone is used to interpret arguments in ISO 8601 format that do not contain time zone information (note that ECMAScript ed 6 draft specifies that date time strings without a time zone are to be treated as local, not UTC)

Your first date format 2015/03/31 is assumed to be March 31st, 2015 at 12am in your current time zone.
Your second date format 2015-03-31 is seen as ISO-8601 and is assumed to be March 31st, 2015 at 12am UTC time zone.

The "Differences in assumed time zone" heading from the linked documentation goes into a bit more detail:

Given a date string of "March 7, 2014", parse() assumes a local time zone, but given an ISO format such as "2014-03-07" it will assume a time zone of UTC. Therefore Date objects produced using those strings will represent different moments in time unless the system is set with a local time zone of UTC. This means that two date strings that appear equivalent may result in two different values depending on the format of the string that is being converted (this behavior is changed in ECMAScript ed 6 so that both will be treated as local).

7
votes

The first string, "2015/03/31", is an unsupported format according to the ECMAScript standard. Whenever an unsupported value is passed to the constructor its behaviour is implementation dependent, i.e. the standard doesn't say what the implementation must do. Some browsers, like Firefox, try to guess what the format is and apparently it creates a date object with that date at midnight local time. Other browsers may return NaN or interpret the parts differently.

The second string, "2015-03-31", is a properly formatted ISO 8601 date. For these strings there are well defined rules and all browsers will interpret it as that date, midnight, UTC.