【问题标题】:SimpleDateFormat is parsing to a different DateSimpleDateFormat 正在解析为不同的日期
【发布时间】:2016-12-26 12:18:21
【问题描述】:

我正在尝试使用 SimpleDateFormat 将字符串解析为日期,但它生成了错误的日期值。

String dataNascimentoValue = "Thu Jan 01 00:00:00 GMT 1970";
SimpleDateFormat converter = new SimpleDateFormat("EEE MMM dd HH:mm:ss zzz yyyy");

Date date = new Date();

converter.setLenient(false);
date = converter.parse( dataNascimentoValue ); //at this moment It generates //a wrong Date: Wed Dec 31 19:00:00 EST 1969

SimpleDateFormat formatter = new SimpleDateFormat("dd/MM/yyyy");
String formatedDate = formatter.format( date );

我已经尝试在 SimpleDateFormat 中设置区域设置,但没有成功。 我想将字符串解析为日期,因为我需要在之后对其进行格式化。

有什么帮助吗?

Screenshot of my try using the Locale.US

【问题讨论】:

  • 您确定屏幕截图中的值吗?使用new Date() 创建的实例如何将 1969 作为年份的日期作为值??
  • 它没有这个值,但是它从以下行获取这个值:“date = converter.parse(dataNascimentoValue);”这是我的问题,在这一行之后,当我使用这个“.parse()”方法时,我的 Date 的值取了一个错误的值。
  • 仅供参考,麻烦的旧日期时间类,如 java.util.Datejava.util.Calendarjava.text.SimpleDateFormat 现在是 legacy,被 Java 8 和 Java 中内置的 java.time 类所取代9. 见Tutorial by Oracle

标签: java date parsing simpledateformat


【解决方案1】:

tl;博士

ZonedDateTime.parse ( 
    "Thu Jan 01 00:00:00 GMT 1970" , 
    DateTimeFormatter.ofPattern ( "EEE MMM dd HH:mm:ss zzz uuuu" , Locale.CANADA )                    
)

功能,而不是错误

这两个字符串:

  • Thur Jan 01 00:00:00 GMT 1970
  • Wed Dec 31 19:00:00 EST 1969

…代表同一时刻,时间轴上的同一点。所以这里没问题;您会看到记录在案的正确行为。

EST 指的是美国和加拿大东部大部分地区使用的时区。此类区域比 UTC (GMT) 晚五个小时。午夜减去五小时是前一天的晚上 7 点(19:00)。不同的wall-clock time,但在宇宙历史上的同一时刻。

在像Date 这样的遗留日期时间类中,许多糟糕的设计选择之一是在调用toString 期间动态应用JVM 的当前默认时区,生成一个值的表示,否则总是在UTC。非常令人困惑,因为这种行为会产生具有时区的错觉,而实际上 Date 始终采用 UTC。更令人困惑的是,实际上在Date 深处有一个时区,用于某些内部目的,但该时区没有任何getter 或setter。这堂课真是一团糟。避免它。

从您的报告中,我可以推断出您的 JVM 在 EST 中有一个当前的默认时区。

使用 java.time

您正在使用麻烦的旧日期时间类,现在是遗留的,被 java.time 类所取代。避免使用这些遗留类;仅使用 java.time 类。

在解析日期时间字符串时始终指定 Locale。如果省略,则 JVM 当前默认的 Locale 用于 (a) 确定人类语言(例如英语)来翻译文本,以及 (b) 确定决定缩写、大写、标点符号等问题的文化规范。

String input = "Thu Jan 01 00:00:00 GMT 1970";
DateTimeFormatter f = DateTimeFormatter.ofPattern ( "EEE MMM dd HH:mm:ss zzz uuuu" , Locale.CANADA );
ZonedDateTime zdt = ZonedDateTime.parse ( input , f );

zdt.toString(): 1970-01-01T00:00Z[GMT]

您可以调整到其他时区。申请ZoneId 以获得另一个ZonedDateTime

continent/region 的格式指定proper time zone name,例如America/MontrealAfrica/CasablancaPacific/Auckland。切勿使用 3-4 个字母的缩写,例如 ESTIST,因为它们不是真正的时区,没有标准化,甚至不是唯一的 (!)。

ZoneId z = ZoneId.of( "America/Montreal" );
ZonedDateTime zdtMontreal = zdt.withZoneSameInstant( z );

zdtMontreal.toString(): 1969-12-31T19:00-05:00[美国/蒙特利尔]

java.time 类上的toString 方法更加合理和清晰,在生成日期时间值的文本表示时使用标准ISO 8601 格式。

live code at IdeOne.com


关于java.time

java.time 框架内置于 Java 8 及更高版本中。这些类取代了麻烦的旧 legacy 日期时间类,例如 java.util.DateCalendarSimpleDateFormat

Joda-Time 项目现在位于maintenance mode,建议迁移到java.time 类。

要了解更多信息,请参阅Oracle Tutorial。并在 Stack Overflow 上搜索许多示例和解释。规格为JSR 310

从哪里获得 java.time 类?

ThreeTen-Extra 项目通过附加类扩展了 java.time。该项目是未来可能添加到 java.time 的试验场。您可以在这里找到一些有用的类,例如IntervalYearWeekYearQuartermore

【讨论】:

  • Basil Bourque,非常感谢您的解释!不幸的是,我的项目是用 Java 6 制作的,有一些如何在不使用新东西的情况下“解决”这种情况?
  • @BrunoMori 仔细阅读我的回答。请特别注意“关于 java.time”部分下的第二个外部项目符号。
【解决方案2】:

您可以使用此方法将字符串解析为日期。

 private static final SimpleDateFormat _simpleDateTimeFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS");

public static Date stringToJavaDateOnly(String dateStr)
            throws ParseException {
        return _simpleDateTimeFormat.parse(dateStr);
    }

【讨论】:

  • 不正确。这种格式甚至比不上输入字符串的格式。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-05-23
  • 1970-01-01
  • 2018-09-17
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多