使用时区将字符串转换为日期

Posted

技术标签:

【中文标题】使用时区将字符串转换为日期【英文标题】:Converting string to date with timezone 【发布时间】:2011-05-11 08:42:50 【问题描述】:

我有一个模式为 yyyy-MM-dd hh:mm a 的字符串 我可以单独获取上面字符串代表日期的时区对象。

我想将其转换为以下格式。 yyyy-MM-dd HH:mm:ss Z

我该怎么做?

【问题讨论】:

joda-time.sourceforge.net 使 java 中的所有日期操作变得更加容易 @Emil 在你说之后我才知道这个功能:),谢谢。 仅供参考,Joda-Time 项目现在位于 maintenance mode,团队建议迁移到 java.time 类。 【参考方案1】:

您可以将SimpleDateFormat 与yyyy-MM-dd HH:mm:ss 一起使用并显式设置TimeZone:

public static Date getSomeDate(final String str, final TimeZone tz)
    throws ParseException 
  final SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd hh:mm a");
  sdf.setTimeZone(tz);
  return sdf.parse(str);


/**
 * @param args
 * @throws IOException
 * @throws InterruptedException
 * @throws ParseException
 */
public static void main(final String[] args) throws ParseException 
  final SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss Z");
  System.out.println(sdf.format(getSomeDate(
      "2010-11-17 01:12 pm", TimeZone.getTimeZone("Europe/Berlin"))));
  System.out.println(sdf.format(getSomeDate(
      "2010-11-17 01:12 pm", TimeZone.getTimeZone("America/Chicago"))));

打印出来:

2010-11-17 13:12:00 +0100

2010-11-17 20:12:00 +0100

2010 年 12 月 1 日更新: 如果要显式打印出特殊的 TimeZone,请将其设置为 SimpleDateFormat:

sdf.setTimeZone(TimeZone .getTimeZone("IST")); 
System.out.println(sdf.format(getSomeDate(
    "2010-11-17 01:12 pm", TimeZone.getTimeZone("IST"))));

打印2010-11-17 13:12:00 +0530

【讨论】:

谢谢迈克尔先生,日期和时间会根据时区更改。记下 +100,它不会更改时区偏移量 @shal:对不起,我不明白你的第二句话。你觉得有什么不对吗? @Michael,我是说时间已正确转换,但看到打印“+0100”的日期值中的“Z”部分并没有改变。例如,如果我正在转换日期从 GMT 到 IST,则转换日期的“Z”部分必须为“+0530”。 这取决于代码运行的系统。对于IST,它会在ideone上打印出2010-11-17 07:42:00 +0000 仅供参考,旧的 CalendarDate 课程现在是 legacy。被java.time 类取代。【参考方案2】:

tl;博士

LocalDateTime.parse(                        // Parse string as value without time zone and without offset-from-UTC.
    "2017-01-23 12:34 PM" , 
    DateTimeFormatter.ofPattern( "uuuu-MM-dd hh:mm a" )
)                                           // Returns a `LocalDateTime` object.
.atZone( ZoneId.of( "America/Montreal" ) )  // Assign time zone, to determine a moment. Returns a `ZonedDateTime` object.
.toInstant()                                // Adjusts from zone to UTC.
.toString()                                 // Generate string: 2017-01-23T17:34:00Z
.replace( "T" , " " )                       // Substitute SPACE for 'T' in middle.
.replace( "Z" , " Z" )                      // Insert SPACE before 'Z'.

避免使用旧的日期时间类

其他答案使用麻烦的旧日期时间类(DateCalendar 等),现在是遗留的,被 java.time 类取代。

LocalDateTime

我有一个模式为 yyyy-MM-dd hh:mm a 的字符串

这样的输入字符串缺少任何与 UTC 或时区偏移的指示。所以我们解析为LocalDateTime

定义格式模式以将您的输入与DateTimeFormatter 对象匹配。

String input = "2017-01-23 12:34 PM" ;
DateTimeFormatter f = DateTimeFormatter.ofPattern( "uuuu-MM-dd hh:mm a" );
LocalDateTime ldt = LocalDateTime.parse( input , f );

ldt.toString(): 2017-01-23T12:34

请注意,LocalDateTime不是特定时刻,只是关于一系列可能时刻的模糊概念。例如,法国巴黎午夜过后的几分钟,在加拿大蒙特利尔仍然是“昨天”。因此,如果没有 Europe/ParisAmerica/Montreal 等时区的上下文,仅仅说“午夜后几分钟”是没有意义的。

ZoneId

我可以单独获取上面字符串代表日期的时区对象。

时区由ZoneId 类表示。

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

ZoneId z = ZoneId.of( "America/Montreal" );

ZonedDateTime

应用ZoneId 得到ZonedDateTime,这确实是时间轴上的一个点,一个特定的历史时刻。

ZonedDateTime zdt = ldt.atZone( z );

zdt.toString(): 2017-01-23T12:34-05:00[美国/蒙特利尔]

Instant

我想将其转换为以下格式。 yyyy-MM-dd HH:mm:ss Z

首先,要知道Z 文字字符是Zulu 的缩写,意思是UTC。换句话说,零小时的offset-from-UTC,+00:00

Instant 类表示UTC 中时间轴上的时刻,分辨率为nanoseconds(最多九 (9) 位小数)。

您可以从ZonedDateTime 中提取Instant 对象。

Instant instant = zdt.toInstant();  // Extracting the same moment but in UTC.

要生成标准ISO 8601 格式的字符串,例如2017-01-22T18:21:13.354Z,请调用toString。标准格式没有空格,使用T 将年-月-日与时-分-秒分开,并规范地附加Z 以获得零偏移量。

String output = instant.toString();

instant.toString(): 2017-01-23T17:34:00Z

我强烈建议尽可能使用标准格式。如果您坚持按照您声明的所需格式使用空格,请在DateTimeFormatter 对象中定义您自己的格式模式,或者只是对Instant::toString 的输出进行字符串操作。

String output = instant.toString()
                       .replace( "T" , " " )  // Substitute SPACE for T.
                       .replace( "Z" , " Z" ); // Insert SPACE before Z.

输出:2017-01-23 17:34:00 Z

试试这个code live 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 类?

Java SE 8SE 9 及更高版本 内置。 标准 Java API 的一部分,带有捆绑实现。 Java 9 添加了一些小功能和修复。 Java SE 6SE 7 大部分 java.time 功能都在ThreeTen-Backport 中向后移植到 Java 6 和 7。 Android ThreeTenABP 项目专门针对 android 改编了 ThreeTen-Backport(如上所述)。 见How to use ThreeTenABP…

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

【讨论】:

缺少 tldr .toInstant() @Muhd 确实。固定的。谢谢!【参考方案3】:

使用SimpleDateFormat

String string1 = "2009-10-10 12:12:12 ";
SimpleDateFormat sdf =  new SimpleDateFormat("yyyy-MM-dd HH:mm:ss Z")
sdf.setTimeZone(tz);
Date date = sdf.parse(string1);

【讨论】:

有时 getID() 会返回类似“America/Argentina/Ushuaia”的字符串。那我该怎么用呢 @Michael Konietzka @shal 我赶紧错过了,更新了答案 @Michael Konietzka 你是对的,但事实并非如此【参考方案4】:

使用您的日期模式创建SimpleDateFormat 的新实例。之后您可以调用它的 parse 方法将日期字符串转换为 java.util.Date 对象。

【讨论】:

【参考方案5】:

毫无疑问,通常使用的格式是2014-10-05T15:23:01Z (TZ)

为此必须使用此代码

SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ");
String dateInString = "2014-10-05T15:23:01Z";

 try 

     Date date = formatter.parse(dateInString.replaceAll("Z$", "+0000"));
     System.out.println(date);

  catch (ParseException e) 
     e.printStackTrace();
 

其输出将是 Sun Oct 05 20:53:01 IST 2014

但是,我不知道为什么我们必须replaceAll "Z" 如果你不添加replaceAll 程序将会失败。

【讨论】:

只是在您的答案中的 replaceAll() 注释上的注释。 replace() 适用于字符串文字,其中 replaceAll() 或 replaceFirst() 支持正则表达式。如果您确实替换(“Z”,“+0000”),那应该给您与您的替换所有(“Z$”,“+0000”)相同的结果【参考方案6】:

请试试这个格式“yyyy-MM-dd HH:mm:ss Z”, 例如:“2020-12-11 22:59:59 GMT”,您可以使用不同的时区,如 PST、GMT 等。

【讨论】:

以上是关于使用时区将字符串转换为日期的主要内容,如果未能解决你的问题,请参考以下文章

JavaScript:将 UTC 日期时间转换为传递时区的日期时间

如何将 UTC 日期时间字符串转换为用户的当前时区?

bigquery 使用时区转换字符串日期时间

如何将包含时区的字符串转换为日期时间 sql server

将输入字符串日期时间从不同时区转换为UTC

Django 自动将日期时间字符串转换为本地时区