使用时区将字符串转换为日期tl;博士避免使用旧日期时间类LocalDateTimeZoneIdZonedDateTimeInstant关于 java.time

2022-09-02 19:39:48

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

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

我该怎么做?


答案 1

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

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-01 更新:如果要显式打印出特殊的时区,请在 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


答案 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'.

避免使用旧日期时间类

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

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

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

ZoneId

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

时区由 ZoneId 类表示。

以 的格式指定适当的时区名称,例如 美国/蒙特利尔非洲/卡萨布兰卡或 。切勿使用3-4个字母的缩写,例如或因为它们不是真正的时区,不是标准化的,甚至不是唯一的(!)。continent/regionPacific/AucklandESTIST

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

ZonedDateTime

应用 以获取一个 ZonedDateTime,它确实是时间轴上的一个点,是历史上的一个特定时刻。ZoneId

ZonedDateTime zdt = ldt.atZone( z );

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

Instant

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

首先,要知道文字字符是 UTC 的缩写,表示 UTC。换言之,与 UTC 的偏移量为零小时。ZZulu+00:00

Instant 类以 UTC 格式表示时间轴上的某个时刻,分辨率为纳秒(最多九 (9) 位小数)。

您可以从 中提取对象。InstantZonedDateTime

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

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

String output = 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

在 IdeOne.com 现场试用此代码。


关于 java.time

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

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

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

从哪里获取 java.time 类?

ThreeTen-Extra 项目通过其他类扩展了 java.time。这个项目是未来可能添加到java.time的试验场。您可以在此处找到一些有用的课程,例如IntervalYearWeekYearQuarter