使用时区将字符串转换为日期
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
。
仅供参考,旧的 Calendar
和 Date
课程现在是 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'.
避免使用旧的日期时间类
其他答案使用麻烦的旧日期时间类(Date
、Calendar
等),现在是遗留的,被 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/Paris
或 America/Montreal
等时区的上下文,仅仅说“午夜后几分钟”是没有意义的。
ZoneId
我可以单独获取上面字符串代表日期的时区对象。
时区由ZoneId
类表示。
以continent/region
的格式指定proper time zone name,例如America/Montreal
、Africa/Casablanca
或Pacific/Auckland
。切勿使用 3-4 个字母的缩写,例如 EST
或 IST
,因为它们不是真正的时区,没有标准化,甚至不是唯一的 (!)。
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.Date
、Calendar
和 SimpleDateFormat
。
Joda-Time 项目现在位于maintenance mode,建议迁移到java.time 类。
要了解更多信息,请参阅Oracle Tutorial。并在 Stack Overflow 上搜索许多示例和解释。规格为JSR 310。
从哪里获得 java.time 类?
Java SE 8 和 SE 9 及更高版本 内置。 标准 Java API 的一部分,带有捆绑实现。 Java 9 添加了一些小功能和修复。 Java SE 6 和 SE 7 大部分 java.time 功能都在ThreeTen-Backport 中向后移植到 Java 6 和 7。 Android ThreeTenABP 项目专门针对 android 改编了 ThreeTen-Backport(如上所述)。 见How to use ThreeTenABP…。ThreeTen-Extra 项目通过附加类扩展了 java.time。该项目是未来可能添加到 java.time 的试验场。您可以在这里找到一些有用的类,例如Interval
、YearWeek
、YearQuarter
和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 等。
【讨论】:
以上是关于使用时区将字符串转换为日期的主要内容,如果未能解决你的问题,请参考以下文章