当我创建一个新的Date对象时,它被初始化为当前时间,但在本地时区。如何获得当前的GMT日期和时间?
当前回答
博士tl;
Instant.now() // Capture the current moment in UTC.
生成一个String来表示该值:
Instant.now().toString()
2016 - 09 - 13 - t23:30:52.123z
细节
正如Jon Skeet的正确答案所述,java.util.Date对象没有时区†。但是它的toString实现在生成日期-时间值的String表示时应用JVM的默认时区。让naïve程序员感到困惑的是,Date似乎有时区,但实际上没有。
java.util。Date、j.u.d alcalendar和Java .text. simpledateformat类与Java捆绑在一起是出了名的麻烦。避免它们。相反,可以使用以下两种有效的日期时间库:
java.time。*包在Java 8 Joda-Time
java。时间(java8)
Java 8带来了一个优秀的新Java .time。*包来取代旧的java.util。日期/日历类。
获取当前UTC/GMT时间是一个简单的一行程序…
Instant instant = Instant.now();
Instant类是java中的基本构建块。time,表示UTC时间轴上的一个时刻,分辨率为纳秒。
在Java 8中,当前时刻的捕获分辨率只有毫秒。Java 9带来了Clock的一个全新实现,它可以捕获当前时刻,最高可达该类的全部纳秒能力,这取决于您的主机时钟硬件的能力。
它的toString方法使用特定的ISO 8601格式生成其值的String表示形式。该格式根据需要输出0、3、6或9位数字(毫秒、微秒或纳秒)来表示秒的分数。
如果您想要更灵活的格式或其他附加特性,那么对于UTC本身应用零的offset-from-UTC (ZoneOffset。UTC常量)来获取OffsetDateTime。
OffsetDateTime now = OffsetDateTime.now( ZoneOffset.UTC );
转储到控制台…
System.out.println( "now.toString(): " + now );
运行时……
now.toString(): 2014-01-21T23:42:03.522Z
关于java.time
java。时间框架内置于Java 8及更高版本中。这些类取代了麻烦的旧遗留日期-时间类,如java.util。日期,日历和简单日期格式。
要了解更多,请参阅Oracle教程。搜索Stack Overflow可以找到很多例子和解释。规范是JSR 310。
Joda-Time项目现在处于维护模式,建议迁移到java。时间类。
你可以交换java。Time对象直接使用数据库。使用符合JDBC 4.2或更高版本的JDBC驱动程序。不需要字符串,不需要java。sql。*类。
从哪里获取java。时间类?
Java SE 8、Java SE 9、Java SE 10、Java SE 11及更高版本——带有捆绑实现的标准Java API的一部分。 Java 9增加了一些小特性并进行了修复。 Java SE 6和Java SE 7 大部分的java。时间功能在ThreeTen-Backport中向后移植到Java 6和7。 安卓 后续版本的Android捆绑实现的java。时间类。 对于早期的Android (<26), ThreeTenABP项目适应ThreeTen-Backport(如上所述)。参见如何使用ThreeTenABP....
ThreeTen-Extra项目扩展了java。额外的课程时间。这个项目是未来可能添加到java.time的一个试验场。你可以在这里找到一些有用的类,比如Interval、YearWeek、YearQuarter等等。
乔达时间
更新:Joda-Time项目现在处于维护模式,建议迁移到java。时间类。
使用第三方开源免费库Joda-Time,只需一行代码就可以获得当前日期-时间。
Joda-Time启发了新的java.time。类,但具有不同的架构。您可以在较旧版本的Java中使用Joda-Time。Joda-Time继续在Java 8中工作,并继续积极维护(截至2014年)。然而,Joda-Time团队建议迁移到java.time。
System.out.println( "UTC/GMT date-time in ISO 8601 format: " + new org.joda.time.DateTime( org.joda.time.DateTimeZone.UTC ) );
更详细的示例代码(Joda-Time 2.3)…
org.joda.time.DateTime now = new org.joda.time.DateTime(); // Default time zone.
org.joda.time.DateTime zulu = now.toDateTime( org.joda.time.DateTimeZone.UTC );
转储到控制台…
System.out.println( "Local time in ISO 8601 format: " + now );
System.out.println( "Same moment in UTC (Zulu): " + zulu );
运行时……
Local time in ISO 8601 format: 2014-01-21T15:34:29.933-08:00
Same moment in UTC (Zulu): 2014-01-21T23:34:29.933Z
有关执行时区工作的更多示例代码,请参阅我对类似问题的回答。
时区
我建议您始终指定一个时区,而不是隐式地依赖JVM当前的默认时区(它随时都可能改变!)。这种依赖似乎是日期时间工作中混乱和错误的常见原因。
调用now()时传递要分配的期望/期望时区。使用DateTimeZone类。
DateTimeZone zoneMontréal = DateTimeZone.forID( "America/Montreal" );
DateTime now = DateTime.now( zoneMontréal );
该类保存UTC时区的一个常量。
DateTime now = DateTime.now( DateTimeZone.UTC );
如果您确实希望使用JVM的当前默认时区,那么可以显式调用,以便代码是自文档化的。
DateTimeZone zoneDefault = DateTimeZone.getDefault();
ISO 8601
阅读ISO 8601格式。java。time和Joda-Time使用该标准的合理格式作为解析和生成字符串的默认值。
†实际上,java.util.Date确实有一个时区,深埋在源代码层之下。对于大多数实际目的,该时区将被忽略。因此,作为速记,我们说java.util.Date没有时区。此外,那个被埋没的时区不是Date的toString方法使用的时区;该方法使用JVM的当前默认时区。因此更有理由避免这个令人困惑的类,并坚持使用Joda-Time和java.time。
其他回答
Calendar c = Calendar.getInstance();
System.out.println("current: "+c.getTime());
TimeZone z = c.getTimeZone();
int offset = z.getRawOffset();
if(z.inDaylightTime(new Date())){
offset = offset + z.getDSTSavings();
}
int offsetHrs = offset / 1000 / 60 / 60;
int offsetMins = offset / 1000 / 60 % 60;
System.out.println("offset: " + offsetHrs);
System.out.println("offset: " + offsetMins);
c.add(Calendar.HOUR_OF_DAY, (-offsetHrs));
c.add(Calendar.MINUTE, (-offsetMins));
System.out.println("GMT Time: "+c.getTime());
博士tl;
Instant.now() // Capture the current moment in UTC.
生成一个String来表示该值:
Instant.now().toString()
2016 - 09 - 13 - t23:30:52.123z
细节
正如Jon Skeet的正确答案所述,java.util.Date对象没有时区†。但是它的toString实现在生成日期-时间值的String表示时应用JVM的默认时区。让naïve程序员感到困惑的是,Date似乎有时区,但实际上没有。
java.util。Date、j.u.d alcalendar和Java .text. simpledateformat类与Java捆绑在一起是出了名的麻烦。避免它们。相反,可以使用以下两种有效的日期时间库:
java.time。*包在Java 8 Joda-Time
java。时间(java8)
Java 8带来了一个优秀的新Java .time。*包来取代旧的java.util。日期/日历类。
获取当前UTC/GMT时间是一个简单的一行程序…
Instant instant = Instant.now();
Instant类是java中的基本构建块。time,表示UTC时间轴上的一个时刻,分辨率为纳秒。
在Java 8中,当前时刻的捕获分辨率只有毫秒。Java 9带来了Clock的一个全新实现,它可以捕获当前时刻,最高可达该类的全部纳秒能力,这取决于您的主机时钟硬件的能力。
它的toString方法使用特定的ISO 8601格式生成其值的String表示形式。该格式根据需要输出0、3、6或9位数字(毫秒、微秒或纳秒)来表示秒的分数。
如果您想要更灵活的格式或其他附加特性,那么对于UTC本身应用零的offset-from-UTC (ZoneOffset。UTC常量)来获取OffsetDateTime。
OffsetDateTime now = OffsetDateTime.now( ZoneOffset.UTC );
转储到控制台…
System.out.println( "now.toString(): " + now );
运行时……
now.toString(): 2014-01-21T23:42:03.522Z
关于java.time
java。时间框架内置于Java 8及更高版本中。这些类取代了麻烦的旧遗留日期-时间类,如java.util。日期,日历和简单日期格式。
要了解更多,请参阅Oracle教程。搜索Stack Overflow可以找到很多例子和解释。规范是JSR 310。
Joda-Time项目现在处于维护模式,建议迁移到java。时间类。
你可以交换java。Time对象直接使用数据库。使用符合JDBC 4.2或更高版本的JDBC驱动程序。不需要字符串,不需要java。sql。*类。
从哪里获取java。时间类?
Java SE 8、Java SE 9、Java SE 10、Java SE 11及更高版本——带有捆绑实现的标准Java API的一部分。 Java 9增加了一些小特性并进行了修复。 Java SE 6和Java SE 7 大部分的java。时间功能在ThreeTen-Backport中向后移植到Java 6和7。 安卓 后续版本的Android捆绑实现的java。时间类。 对于早期的Android (<26), ThreeTenABP项目适应ThreeTen-Backport(如上所述)。参见如何使用ThreeTenABP....
ThreeTen-Extra项目扩展了java。额外的课程时间。这个项目是未来可能添加到java.time的一个试验场。你可以在这里找到一些有用的类,比如Interval、YearWeek、YearQuarter等等。
乔达时间
更新:Joda-Time项目现在处于维护模式,建议迁移到java。时间类。
使用第三方开源免费库Joda-Time,只需一行代码就可以获得当前日期-时间。
Joda-Time启发了新的java.time。类,但具有不同的架构。您可以在较旧版本的Java中使用Joda-Time。Joda-Time继续在Java 8中工作,并继续积极维护(截至2014年)。然而,Joda-Time团队建议迁移到java.time。
System.out.println( "UTC/GMT date-time in ISO 8601 format: " + new org.joda.time.DateTime( org.joda.time.DateTimeZone.UTC ) );
更详细的示例代码(Joda-Time 2.3)…
org.joda.time.DateTime now = new org.joda.time.DateTime(); // Default time zone.
org.joda.time.DateTime zulu = now.toDateTime( org.joda.time.DateTimeZone.UTC );
转储到控制台…
System.out.println( "Local time in ISO 8601 format: " + now );
System.out.println( "Same moment in UTC (Zulu): " + zulu );
运行时……
Local time in ISO 8601 format: 2014-01-21T15:34:29.933-08:00
Same moment in UTC (Zulu): 2014-01-21T23:34:29.933Z
有关执行时区工作的更多示例代码,请参阅我对类似问题的回答。
时区
我建议您始终指定一个时区,而不是隐式地依赖JVM当前的默认时区(它随时都可能改变!)。这种依赖似乎是日期时间工作中混乱和错误的常见原因。
调用now()时传递要分配的期望/期望时区。使用DateTimeZone类。
DateTimeZone zoneMontréal = DateTimeZone.forID( "America/Montreal" );
DateTime now = DateTime.now( zoneMontréal );
该类保存UTC时区的一个常量。
DateTime now = DateTime.now( DateTimeZone.UTC );
如果您确实希望使用JVM的当前默认时区,那么可以显式调用,以便代码是自文档化的。
DateTimeZone zoneDefault = DateTimeZone.getDefault();
ISO 8601
阅读ISO 8601格式。java。time和Joda-Time使用该标准的合理格式作为解析和生成字符串的默认值。
†实际上,java.util.Date确实有一个时区,深埋在源代码层之下。对于大多数实际目的,该时区将被忽略。因此,作为速记,我们说java.util.Date没有时区。此外,那个被埋没的时区不是Date的toString方法使用的时区;该方法使用JVM的当前默认时区。因此更有理由避免这个令人困惑的类,并坚持使用Joda-Time和java.time。
SimpleDateFormat dateFormatGmt = new SimpleDateFormat("yyyy-MM-dd");
dateFormatGmt.setTimeZone(TimeZone.getTimeZone("GMT"));
System.out.println(dateFormatGmt.format(date));
以特定时区和特定格式呈现系统时间的示例代码。
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.TimeZone;
public class TimZoneTest {
public static void main (String[] args){
//<GMT><+/-><hour>:<minutes>
// Any screw up in this format, timezone defaults to GMT QUIETLY. So test your format a few times.
System.out.println(my_time_in("GMT-5:00", "MM/dd/yyyy HH:mm:ss") );
System.out.println(my_time_in("GMT+5:30", "'at' HH:mm a z 'on' MM/dd/yyyy"));
System.out.println("---------------------------------------------");
// Alternate format
System.out.println(my_time_in("America/Los_Angeles", "'at' HH:mm a z 'on' MM/dd/yyyy") );
System.out.println(my_time_in("America/Buenos_Aires", "'at' HH:mm a z 'on' MM/dd/yyyy") );
}
public static String my_time_in(String target_time_zone, String format){
TimeZone tz = TimeZone.getTimeZone(target_time_zone);
Date date = Calendar.getInstance().getTime();
SimpleDateFormat date_format_gmt = new SimpleDateFormat(format);
date_format_gmt.setTimeZone(tz);
return date_format_gmt.format(date);
}
}
输出
10/08/2011 21:07:21
at 07:37 AM GMT+05:30 on 10/09/2011
at 19:07 PM PDT on 10/08/2011
at 23:07 PM ART on 10/08/2011
这为我工作,返回格林尼治时间戳!
Date currDate;
SimpleDateFormat dateFormatGmt = new SimpleDateFormat("yyyy-MMM-dd HH:mm:ss");
dateFormatGmt.setTimeZone(TimeZone.getTimeZone("GMT"));
SimpleDateFormat dateFormatLocal = new SimpleDateFormat("yyyy-MMM-dd HH:mm:ss");
long currTime = 0;
try {
currDate = dateFormatLocal.parse( dateFormatGmt.format(new Date()) );
currTime = currDate.getTime();
} catch (ParseException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
推荐文章
- 到底是什么导致了堆栈溢出错误?
- 为什么Android工作室说“等待调试器”如果我不调试?
- 是否有可能更新一个本地化的故事板的字符串?
- Java:路径vs文件
- ExecutorService,如何等待所有任务完成
- 为什么在JavaScript的Date构造函数中month参数的范围从0到11 ?
- Maven依赖Servlet 3.0 API?
- 如何在IntelliJ IDEA中添加目录到应用程序运行概要文件中的类路径?
- getter和setter是糟糕的设计吗?相互矛盾的建议
- Android room persistent: AppDatabase_Impl不存在
- Java的String[]在Kotlin中等价于什么?
- Intellij IDEA上的System.out.println()快捷方式
- 使用Spring RestTemplate获取JSON对象列表
- Spring JPA选择特定的列
- URLEncoder不能翻译空格字符