如何在JavaScript中使用格式规范将字符串转换为日期时间?

230

我该如何在JavaScript中通过指定格式字符串将一个字符串转换为日期时间对象?

我想找到这样的方法:

var dateTime = convertToDateTime("23.11.2009 12:34:56", "dd.MM.yyyy HH:mm:ss");

2
顺便提一下,点和连字符作为日期分隔符可能会失败,斜杠可以使用,因此(根据我的测试)Javascript将接受“2011/08/22 10:30:00”,但尚未(尽管ISO 8601)支持“2011-09-02 15:58:40”,据称在Javascript 1.8.5上支持。 - Dave Everitt
Date.parse函数会解析日期字符串,该字符串的格式应为“mm/dd/yyyy”。在应用Parse之前,请将字符串转换为“mm/dd/yyyy”格式。 - user958894
为什么不按所需的格式提供日期呢?我会这样做:new Date('2012 11 25 18:00:00');它可以工作! - foxybagga
15个回答

110

如果您的字符串兼容Date.parse(),请使用new Date(dateString)。如果您的格式不兼容(我认为不兼容),则需要自己解析字符串(使用正则表达式应该很容易),并创建一个新的日期对象,为年、月、日、小时、分钟和秒显式指定值。


2
Date.parse是最好的解决方案!它几乎可以解析任何东西!+1 - ianaz
13
几乎所有在美国使用的东西,但大部分欧洲使用的格式缺失是一个相当大的“几乎”。不过,新的日期格式是自定义函数的良好基础。 - Pavel V.
new Date(dateString) 对我来说运行良好,而 Date.parse() 则将其转换为毫秒并返回数字而不是日期类型。 - Reagan Ochora

86

8
类型错误:对象日期(Date)没有“getDateFromFormat”方法。 - Derek 朕會功夫
1
+1,好的库...纯JavaScript...太棒了。谢谢。 - Surendra Jnawali
12
两个链接已失效。 - Sytham
1
如果可以的话,请更新链接。这已经没有帮助了。 - Tallerlei
1
不幸的是,javascripttoolbox.com 链接也无法访问。 - vchan

73

@Christoph提到使用正则表达式来解决问题。这是我正在使用的:

var dateString = "2010-08-09 01:02:03";
var reggie = /(\d{4})-(\d{2})-(\d{2}) (\d{2}):(\d{2}):(\d{2})/;
var dateArray = reggie.exec(dateString); 
var dateObject = new Date(
    (+dateArray[1]),
    (+dateArray[2])-1, // Careful, month starts at 0!
    (+dateArray[3]),
    (+dateArray[4]),
    (+dateArray[5]),
    (+dateArray[6])
);

这绝不是智能的,只需配置正则表达式和new Date(blah)以满足您的需求。

编辑:在ES6中使用解构可能更容易理解:

let dateString = "2010-08-09 01:02:03"
  , reggie = /(\d{4})-(\d{2})-(\d{2}) (\d{2}):(\d{2}):(\d{2})/
  , [, year, month, day, hours, minutes, seconds] = reggie.exec(dateString)
  , dateObject = new Date(year, month-1, day, hours, minutes, seconds);

但说实话,现在我会考虑使用类似 Moment 的东西


19
@bažmegakapa,你可以尝试编辑答案并自己添加 'var',而不是哭泣。 - OrganicPanda
1
@minitech /g 是从一个现有项目中粗制滥造的复制/粘贴。lastindex 是由编辑添加的(为了对抗 \g 吧?)。我现在会将它们移除。感谢您指出这一点。 - OrganicPanda
1
非常好的答案..对我真的很有用 - Jethik
有关如何将时区添加到其中的任何评论? - Sagar Ranglani
@SagarRanglani 说实话,现在最好使用一个库来处理这个问题,这个答案在2011年还可以,但现在我会使用date-fns https://date-fns.org 或类似的库。 - OrganicPanda
显示剩余2条评论

15

JavaScript没有复杂的日期/时间格式化程序。

你将需要使用外部库来格式化日期输出。Flagrant Badassery的"JavaScript Date Format"看起来非常有前途。

关于输入转换,已经提出了几个建议。 :)


3
“have to”是相对的。当然,你可以构建自己的实现方式。 :) - Tomalak

13

看看Moment.js。它是一个现代且功能强大的库,可以弥补JavaScript中糟糕的日期函数(或缺乏)的不足。


5
我没有给这篇文章点踩,但也许知道Moment.js中哪个函数可以提供所需的行为会更好(对于我们这些已经在使用Moment.js但不知道具体使用哪个函数的懒惰人来说)。 - Matt
本地化看起来很有前途,选择任何中欧区域设置以获取点格式。但是,我没有测试过,并且完整的日期时间格式不在我看到的示例中。 - Pavel V.
1
我没有点踩,但我想说这个很慢。对于桌面来说还好,但对于移动设备可能会太慢了(取决于你的需求)。 - gabn88

12

这里更新一个答案,有一个不错的js库在http://www.datejs.com/

Datejs是一个开源的JavaScript日期库,可用于解析、格式化和处理日期。


回滚了那个编辑,虽然非常有帮助,但应该作为评论或单独的答案添加。 - Alexis Abril
这段内容太长了,不适合作为评论,而且添加重复的答案也是不正确的。这个问题已经有超过155,000次的浏览量 - 让我们尽可能地为这些人提供更多信息!(顺便说一句 - 在Stack Overflow上编辑现有答案非常普遍 - 如果我只是复制了你的答案,你可能会感到更糟糕...) - Kobi
1
无论如何,如果您不喜欢我的编辑,那没关系,请扩展您的答案 - “仅供更新答案”是填充语,缺少一些细节。顺便说一句,这个问题目前正在引起一些关注,这就是您突然获得5票的原因。 - Kobi

7
var temp1 = "";
var temp2 = "";

var str1 = fd; 
var str2 = td;

var dt1  = str1.substring(0,2);
var dt2  = str2.substring(0,2);

var mon1 = str1.substring(3,5);
var mon2 = str2.substring(3,5);

var yr1  = str1.substring(6,10);  
var yr2  = str2.substring(6,10); 

temp1 = mon1 + "/" + dt1 + "/" + yr1;
temp2 = mon2 + "/" + dt2 + "/" + yr2;

var cfd = Date.parse(temp1);
var ctd = Date.parse(temp2);

var date1 = new Date(cfd); 
var date2 = new Date(ctd);

if(date1 > date2) { 
    alert("FROM DATE SHOULD BE MORE THAN TO DATE");
}

7
time = "2017-01-18T17:02:09.000+05:30"

t = new Date(time)

hr = ("0" + t.getHours()).slice(-2);
min = ("0" + t.getMinutes()).slice(-2);
sec = ("0" + t.getSeconds()).slice(-2);

t.getFullYear()+"-"+t.getMonth()+1+"-"+t.getDate()+" "+hr+":"+min+":"+sec

在调用getmonth()函数后加1似乎是错误的。 - Arjunsingh
1
t.getMonth()+1 是正确的,因为 Date.getMonth() 返回基于0的值。 - Jussi Palo

6

对于解析一个或两个日期来说,使用外部库就有点大材小用了。因此,我使用OliChristoph的解决方案编写了自己的函数。在中欧,我们很少使用除OP格式以外的任何格式,因此这应该足够简单应用程序的需求。

function ParseDate(dateString) {
    //dd.mm.yyyy, or dd.mm.yy
    var dateArr = dateString.split(".");
    if (dateArr.length == 1) {
        return null;    //wrong format
    }
    //parse time after the year - separated by space
    var spacePos = dateArr[2].indexOf(" ");
    if(spacePos > 1) {
        var timeString = dateArr[2].substr(spacePos + 1);
        var timeArr = timeString.split(":");
        dateArr[2] = dateArr[2].substr(0, spacePos);
        if (timeArr.length == 2) {
            //minutes only
            return new Date(parseInt(dateArr[2]), parseInt(dateArr[1]-1), parseInt(dateArr[0]), parseInt(timeArr[0]), parseInt(timeArr[1]));
        } else {
            //including seconds
            return new Date(parseInt(dateArr[2]), parseInt(dateArr[1]-1), parseInt(dateArr[0]), parseInt(timeArr[0]), parseInt(timeArr[1]), parseInt(timeArr[2]))
        }
    } else {
        //gotcha at months - January is at 0, not 1 as one would expect
        return new Date(parseInt(dateArr[2]), parseInt(dateArr[1] - 1), parseInt(dateArr[0]));
    }
}

4

Date.parse() 相当智能,但我不能保证它可以正确解析格式。

如果无法解析,您需要找到一些方法来连接两个部分。您的示例非常简单(仅由数字组成),因此使用一点正则表达式(或甚至是 string.split() -- 可能更快)配合一些 parseInt() 就可以快速创建日期。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接