在Java中输出RFC 3339时间戳

时间:2008-11-14 05:26:06

标签: java timestamp rfc3339

我想输出带有PST偏移的时间戳(例如,2008-11-13T13:23:30-08:00)。 java.util.SimpleDateFormat似乎没有以小时:分钟格式输出时区偏移量,它会排除冒号。有一种简单的方法可以在Java中获取时间戳吗?

// I want 2008-11-13T12:23:30-08:00
String timestamp = new SimpleDateFormat("yyyy-MM-dd'T'h:m:ssZ").format(new Date());
System.out.println(timestamp); 
// prints "2008-11-13T12:23:30-0800" See the difference?

此外,SimpleDateFormat无法正确解析上述示例。它会抛出ParseException

// Throws a ParseException
new SimpleDateFormat("yyyy-MM-dd'T'h:m:ssZ").parse("2008-11-13T13:23:30-08:00")

12 个答案:

答案 0 :(得分:40)

从Java 7开始,ISO8601时区的X模式字符串。对于您描述的格式的字符串,请使用XXXSee the documentation

样品:

System.out.println(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX")
        .format(new Date()));

结果:

2014-03-31T14:11:29+02:00

答案 1 :(得分:16)

查看Joda Time包。它们使RFC 3339的日期格式化变得更加容易。

Joda示例:

DateTime dt = new DateTime(2011,1,2,12,45,0,0, DateTimeZone.UTC);
DateTimeFormatter fmt = ISODateTimeFormat.dateTime();
String outRfc = fmt.print(dt);

答案 2 :(得分:12)

从“get it done dept”开始,一个解决方案就是在SimpleDateFormat完成后使用正则表达式修复字符串。类似s /(\ d {2})(\ d {2})$ / $ 1:$ 2 / Perl。

如果您对此感兴趣,我将使用可用的Java代码编辑此响应。

但是,是的。我也遇到了这个问题。 RFC3339,我在看着你!

编辑:

这对我有用

// As a private class member
private SimpleDateFormat rfc3339 = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ");

String toRFC3339(Date d)
{
   return rfc3339.format(d).replaceAll("(\\d\\d)(\\d\\d)$", "$1:$2");
}

答案 3 :(得分:11)

我花了很多时间寻找同一问题的答案,我在这里找到了一些东西:http://developer.android.com/reference/java/text/SimpleDateFormat.html

建议的答案:

String timestamp = new SimpleDateFormat("yyyy-MM-dd'T'h:m:ssZZZZZ").format(new Date());

如果您发现我正在使用5' Z'而不是一个。这使得输出在偏移量中具有冒号,如下所示:" 2008-11-13T12:23:30-08:00"。希望它有所帮助。

答案 4 :(得分:4)

问题是Z生成时区偏移而没有冒号(:)作为分隔符。

答案 5 :(得分:3)

SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd'T'h:m:ss.SZ");

不是你需要什么?

答案 6 :(得分:1)

我找到了一个帮助我解决问题的迷路PasteBin:http://pastebin.com/y3TCAikc

以防其内容后来被删除:

// I want 2008-11-13T12:23:30-08:00
String timestamp = new SimpleDateFormat("yyyy-MM-dd'T'h:m:ssZ").format(new Date());
System.out.println(timestamp); 
// prints "2008-11-13T12:23:30-0800" See the difference?

// Throws a ParseException
new SimpleDateFormat("yyyy-MM-dd'T'h:m:ssZ").parse("2008-11-13T13:23:30-08:00")

SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd'T'h:m:ss.SZ");

答案 7 :(得分:1)

我为RFC3339创建了一个InternetDateFormat类。

但源代码评论是日文。

PS:我创建了英文版并重构了一点。

答案 8 :(得分:1)

为此,我们可以简单地使用ZonedDateTime classDateTimeFormatter class

DateTimeFormatter format = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ssxxx");
ZonedDateTime z2 = ZonedDateTime.now(ZoneOffset.UTC).truncatedTo(ChronoUnit.SECONDS);
System.out.println("format =======> " + z2.format(format));

输出:格式=======> 30-03-2020T05:57:37 + 00:00

答案 9 :(得分:1)

java.time

java.util 日期时间 API 及其格式化 API SimpleDateFormat 已过时且容易出错。建议完全停止使用它们并切换到 modern Date-Time API*

另外,下面引用的是来自 home page of Joda-Time 的通知:

<块引用>

请注意,从 Java SE 8 开始,要求用户迁移到 java.time (JSR-310) - JDK 的核心部分,取代了该项目。

使用 java.time(现代日期时间 API)的解决方案:Pacific Time Zone 中最大的城市是洛杉矶,其时区名称为 America/Los_Angeles em>。使用 ZoneId.of("America/Los_Angeles"),您可以创建一个 ZonedDateTime 实例,该实例旨在在 DST 转换时自动调整时区偏移。

如果您需要时区偏移量而不是时区名称,您可以使用 ZonedDateTime#toOffsetDateTimeZonedDateTime 转换为 OffsetDateTimeOffsetDateTime 的其他一些用途是创建具有固定时区偏移量的 Date-Time 实例(例如 Instant.now().atOffset(ZoneOffset.of("+05:30")),并解析具有时区偏移量的 Date-Time 字符串。

演示:

import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.OffsetDateTime;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.time.temporal.ChronoUnit;

public class Main {
    public static void main(String[] args) {
        ZoneId zoneIdLosAngeles = ZoneId.of("America/Los_Angeles");
        ZonedDateTime zdtNowLosAngeles = ZonedDateTime.now(zoneIdLosAngeles);
        System.out.println(zdtNowLosAngeles);

        // With zone offset but without time zone name
        OffsetDateTime odtNowLosAngeles = zdtNowLosAngeles.toOffsetDateTime();
        System.out.println(odtNowLosAngeles);

        // Truncated up to seconds
        odtNowLosAngeles = odtNowLosAngeles.truncatedTo(ChronoUnit.SECONDS);
        System.out.println(odtNowLosAngeles);

        // ################ A winter date-time ################
        ZonedDateTime zdtLosAngelesWinter = ZonedDateTime
                .of(LocalDateTime.of(LocalDate.of(2021, 11, 20), LocalTime.of(10, 20)), zoneIdLosAngeles);
        System.out.println(zdtLosAngelesWinter); // 2021-11-20T10:20-08:00[America/Los_Angeles]
        System.out.println(zdtLosAngelesWinter.toOffsetDateTime()); // 2021-11-20T10:20-08:00

        // ################ Parsing a date-time string with zone offset ################
        String strDateTime = "2008-11-13T13:23:30-08:00";
        OffsetDateTime odt = OffsetDateTime.parse(strDateTime);
        System.out.println(odt); // 2008-11-13T13:23:30-08:00
    }
}

样本运行的输出:

2021-07-18T03:27:15.578028-07:00[America/Los_Angeles]
2021-07-18T03:27:15.578028-07:00
2021-07-18T03:27:15-07:00
2021-11-20T10:20-08:00[America/Los_Angeles]
2021-11-20T10:20-08:00
2008-11-13T13:23:30-08:00

ONLINE DEMO

您一定已经注意到我没有使用 DateTimeFormatter 来解析您问题的日期时间字符串。这是因为您的日期时间字符串符合 ISO-8601 标准。现代日期时间 API 基于 ISO 8601,只要日期时间字符串符合 ISO 8601 标准,就不需要明确使用 DateTimeFormatter 对象。

Trail: Date Time 了解有关现代 Date-Time API 的更多信息。


* 出于任何原因,如果您必须坚持使用 Java 6 或 Java 7,您可以使用 ThreeTen-Backport,它将大部分 java.time 功能向后移植到 Java 6 & 7. 如果您正在为 Android 项目工作并且您的 Android API 级别仍然不符合 Java-8,请检查 Java 8+ APIs available through desugaringHow to use ThreeTenABP in Android Project

答案 10 :(得分:0)

我尝试过这种格式并为我工作yyyy-MM-dd'T'HH:mm:ss'Z'

答案 11 :(得分:0)

我用这个测试了很多,对我来说效果很好......特别是在解析(以及格式化)方面,它是我迄今为止发现的最接近的

DateTimeFormatter rfc3339Formatter = DateTimeFormatter.ISO_OFFSET_DATE_TIME;

DateTimeFormatter rfc3339Parser = new DateTimeFormatterBuilder()
    .parseCaseInsensitive()
    .appendValue(ChronoField.YEAR, 4)
    .appendLiteral('-')
    .appendValue(ChronoField.MONTH_OF_YEAR, 2)
    .appendLiteral('-')
    .appendValue(ChronoField.DAY_OF_MONTH, 2)
    .appendLiteral('T')
    .appendValue(ChronoField.HOUR_OF_DAY, 2)
    .appendLiteral(':')
    .appendValue(ChronoField.MINUTE_OF_HOUR, 2)
    .appendLiteral(':')
    .appendValue(ChronoField.SECOND_OF_MINUTE, 2)
    .optionalStart()
    .appendFraction(ChronoField.NANO_OF_SECOND, 2, 9, true) //2nd parameter: 2 for JRE (8, 11 LTS), 1 for JRE (17 LTS)
    .optionalEnd()
    .appendOffset("+HH:MM","Z")
    .toFormatter()
    .withResolverStyle(ResolverStyle.STRICT)
    .withChronology(IsoChronology.INSTANCE);

https://github.com/guyplusplus/RFC3339-DateTimeFormatter 处的测试用例