将日期字符串解析为java.util.Date时出现非法模式字符"T"

hgu*_*ser 164 java date iso8601 simpledateformat

我有一个日期字符串,我想解析它到正常日期使用java Date API,以下是我的代码:

public static void main(String[] args) {
    String date="2010-10-02T12:23:23Z";
    String pattern="yyyy-MM-ddThh:mm:ssZ";
    SimpleDateFormat sdf=new SimpleDateFormat(pattern);
    try {
        Date d=sdf.parse(date);
        System.out.println(d.getYear());
    } catch (ParseException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}
Run Code Online (Sandbox Code Playgroud)

但是我有一个例外: java.lang.IllegalArgumentException: Illegal pattern character 'T'

所以我想知道我是否必须拆分字符串并手动解析它?

顺便说一句,我试图在T的两边添加单引号字符:

String pattern="yyyy-MM-dd'T'hh:mm:ssZ";
Run Code Online (Sandbox Code Playgroud)

它也行不通.

小智 190

Java 8及更高版本的更新

您现在可以简单地执行Instant.parse("2015-04-28T14:23:38.521Z")并获取正确的内容,尤其是因为您应该使用Java Instant而不是使用java.util.Date最新版本的Java.

你应该使用DateTimeFormatter而不是SimpleDateFormatter.

原答案:

下面的解释仍然有效,因为格式代表什么.但它是在Java 8无处不在之前编写的,所以如果你使用Java 8或更高版本,它会使用你不应该使用的旧类.

这适用于带有尾随的输入,Z如下所示:

在模式中,任何一方T都逃脱了'.

最后的模式Z实际上XXX是在JavaDoc中记录的SimpleDateFormat,实际上如何使用它并不是很清楚,因为它Z也是旧TimeZone信息的标记 .

Q2597083.java

import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.TimeZone;

public class Q2597083
{
    /**
     * All Dates are normalized to UTC, it is up the client code to convert to the appropriate TimeZone.
     */
    public static final TimeZone UTC;

    /**
     * @see <a href="http://en.wikipedia.org/wiki/ISO_8601#Combined_date_and_time_representations">Combined Date and Time Representations</a>
     */
    public static final String ISO_8601_24H_FULL_FORMAT = "yyyy-MM-dd'T'HH:mm:ss.SSSXXX";

    /**
     * 0001-01-01T00:00:00.000Z
     */
    public static final Date BEGINNING_OF_TIME;

    /**
     * 292278994-08-17T07:12:55.807Z
     */
    public static final Date END_OF_TIME;

    static
    {
        UTC = TimeZone.getTimeZone("UTC");
        TimeZone.setDefault(UTC);
        final Calendar c = new GregorianCalendar(UTC);
        c.set(1, 0, 1, 0, 0, 0);
        c.set(Calendar.MILLISECOND, 0);
        BEGINNING_OF_TIME = c.getTime();
        c.setTime(new Date(Long.MAX_VALUE));
        END_OF_TIME = c.getTime();
    }

    public static void main(String[] args) throws Exception
    {

        final SimpleDateFormat sdf = new SimpleDateFormat(ISO_8601_24H_FULL_FORMAT);
        sdf.setTimeZone(UTC);
        System.out.println("sdf.format(BEGINNING_OF_TIME) = " + sdf.format(BEGINNING_OF_TIME));
        System.out.println("sdf.format(END_OF_TIME) = " + sdf.format(END_OF_TIME));
        System.out.println("sdf.format(new Date()) = " + sdf.format(new Date()));
        System.out.println("sdf.parse(\"2015-04-28T14:23:38.521Z\") = " + sdf.parse("2015-04-28T14:23:38.521Z"));
        System.out.println("sdf.parse(\"0001-01-01T00:00:00.000Z\") = " + sdf.parse("0001-01-01T00:00:00.000Z"));
        System.out.println("sdf.parse(\"292278994-08-17T07:12:55.807Z\") = " + sdf.parse("292278994-08-17T07:12:55.807Z"));
    }
}
Run Code Online (Sandbox Code Playgroud)

产生以下输出:

sdf.format(BEGINNING_OF_TIME) = 0001-01-01T00:00:00.000Z
sdf.format(END_OF_TIME) = 292278994-08-17T07:12:55.807Z
sdf.format(new Date()) = 2015-04-28T14:38:25.956Z
sdf.parse("2015-04-28T14:23:38.521Z") = Tue Apr 28 14:23:38 UTC 2015
sdf.parse("0001-01-01T00:00:00.000Z") = Sat Jan 01 00:00:00 UTC 1
sdf.parse("292278994-08-17T07:12:55.807Z") = Sun Aug 17 07:12:55 UTC 292278994
Run Code Online (Sandbox Code Playgroud)


Bas*_*que 24

TL;博士

Instant.parse( "2010-10-02T12:23:23Z" )
Run Code Online (Sandbox Code Playgroud)

ISO 8601

该格式由日期时间字符串格式的ISO 8601标准定义.

都:

...默认情况下使用ISO 8601格式来解析和生成字符串.

您通常应该避免使用旧的java.util.Date /.Calendar和java.text.SimpleDateFormat类,因为它们非常麻烦,令人困惑和有缺陷.如果需要进行互操作,您可以来回转换.

java.time

内置于Java 8及更高版本的是新的java.time框架.灵感来自Joda-Time,由JSR 310定义,并由ThreeTen-Extra项目扩展.

Instant instant = Instant.parse( "2010-10-02T12:23:23Z" );  // `Instant` is always in UTC.
Run Code Online (Sandbox Code Playgroud)

转换为旧类.

java.util.Date date = java.util.Date.from( instant );  // Pass an `Instant` to the `from` method.
Run Code Online (Sandbox Code Playgroud)

时区

如果需要,您可以指定时区.

ZoneId zoneId = ZoneId.of( "America/Montreal" ); // Define a time zone rather than rely implicitly on JVM’s current default time zone.
ZonedDateTime zdt = ZonedDateTime.ofInstant( instant , zoneId );  // Assign a time zone adjustment from UTC.
Run Code Online (Sandbox Code Playgroud)

兑换.

java.util.Date date = java.util.Date.from( zdt.toInstant() );  // Extract an `Instant` from the `ZonedDateTime` to pass to the `from` method.
Run Code Online (Sandbox Code Playgroud)

乔达时间

更新:Joda-Time项目现在处于维护模式.该团队建议迁移到java.time类.

以下是Joda-Time 2.8中的一些示例代码.

org.joda.time.DateTime dateTime_Utc = new DateTime( "2010-10-02T12:23:23Z" , DateTimeZone.UTC );  // Specifying a time zone to apply, rather than implicitly assigning the JVM’s current default.
Run Code Online (Sandbox Code Playgroud)

转换为旧班.请注意,转换时会丢失指定的时区,因为无法为juDate分配时区.

java.util.Date date = dateTime_Utc.toDate(); // The `toDate` method converts to old class.
Run Code Online (Sandbox Code Playgroud)

时区

如果需要,您可以指定时区.

DateTimeZone zone = DateTimeZone.forID( "America/Montreal" );
DateTime dateTime_Montreal = dateTime_Utc.withZone ( zone );
Run Code Online (Sandbox Code Playgroud)

关于java.time

java.time框架是建立在Java 8和更高版本.这些类取代麻烦的老传统日期时间类,如java.time.Instant,java.util.Date,和Calendar.

现在处于维护模式Joda-Time项目建议迁移到java.time类.

要了解更多信息,请参阅Oracle教程.并搜索Stack Overflow以获取许多示例和解释.规范是JSR 310.

您可以直接与数据库交换java.time对象.使用符合JDBC 4.2或更高版本的JDBC驱动程序.不需要字符串,不需要课程.SimpleDateFormat

从哪里获取java.time类?

ThreeTen-额外项目与其他类扩展java.time.该项目是未来可能添加到java.time的试验场.您可以在此比如找到一些有用的类java.sql.*,Interval,YearWeek,和更多.