如何在Java中从日期对象格式中删除毫秒

22

由于java.util.Date对象将日期存储为2014-01-24 17:33:47.214,但我希望将日期格式化为2014-01-24 17:33:47我想去掉毫秒部分。

我查看了一个与我的问题相关的问题...

如何删除Date对象的子秒部分

我尝试了给出的答案

long time = date.getTime();
date.setTime((time / 1000) * 1000);

但是我的日期格式为2014-01-24 17:33:47.0,我该如何去掉这个日期格式中的0


2
日期不重要,显示您的格式化代码。 - Matt Ball
1
一个日期对象代表长的毫秒数,你不能将它除以1000并仅显示小时...也许你需要查阅文档。 - sanbhat
我相信他们的意思是使用 date.setTime(Math.round(time / 1000) * 1000);(请参见下面的答案)。 - Nick Grealy
@NickG:鉴于time / 1000始终是一个整数,四舍五入的意义何在? - Jon Skeet
时间值是一个长整型,因此没有舍入的必要。 - Jason
@Jason,太对了。这是一个长的。谢谢。 - Nick Grealy
8个回答

35

简而言之

去掉小数秒。

myJavaUtilDate.toInstant()                         // Convert from legacy class to modern class. Returns a `Instant` object.
              .truncatedTo( ChronoUnit.SECONDS )   // Generate new `Instant` object based on the values of the original, but chopping off the fraction-of-second.

隐藏小数秒,生成一个String时。
myJavaUtilDate.toInstant()                         // Convert from legacy class to modern class. Returns a `Instant` object.
              .atOffset( ZoneOffset.UTC )          // Return a `OffsetDateTime` object. 
              .format( DateTimeFormatter.ofPattern( "uuuu-MM-dd HH:mm:ss" ) ). // Ask the `OffsetDateTime` object to generate a `String` with text representing its value, in a format defined in the `DateTimeFormatter` object.

避免使用过时的日期时间类

您正在使用麻烦的旧日期时间类,现在已经被java.time类取代。

Instant

通过调用添加到旧类中的新方法,将您的旧java.util.Date对象转换为java.time.Instant

Instant instant = myJavaUtilDate.toInstant() ;

Table of types of date-time classes in modern java.time versus legacy.

截取

如果您想要更改数据本身的值以删除一部分秒数,可以使用截取功能。Java.time类使用不可变对象,因此我们生成新对象而不是修改(突变)原始数据。

Instant instantTruncated = instant.truncatedTo( ChronoUnit.SECONDS );

生成字符串

如果您想要生成表示日期时间值的字符串时,而不是截断小数秒,则可以定义适合您需要的格式化程序来抑制小数秒的显示。

例如,“uuuu-MM-dd HH:mm:ss”没有提到小数秒,因此数据中包含的任何毫秒都不会出现在生成的字符串中。

Instant转换为OffsetDateTime以进行更灵活的格式化。

DateTimeFormatter f = DateTimeFormatter.ofPattern( "uuuu-MM-dd HH:mm:ss" );
OffsetDateTime odt = instant.atOffset( ZoneOffset.UTC )
String output = odt.format( f );

时区

请注意,您的问题忽略了时区的问题。如果您打算使用UTC,则上述代码按定义在UTC中同时适用于DateInstant。如果您想通过某个地区的时钟时间来观察给定数据,请应用时区。搜索Stack Overflow以获取更多关于ZoneIdZonedDateTime类名称的信息。


关于java.time

java.time框架内置于Java 8及以后版本。这些类替代了老旧的遗留日期时间类,如java.util.DateCalendarSimpleDateFormat

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

想了解更多,请参阅Oracle教程。并在Stack Overflow上搜索许多示例和解释。规范是JSR 310

您可以直接与数据库交换java.time对象。使用符合JDBC 4.2或更高版本的JDBC驱动程序。不需要字符串,也不需要java.sql.*类。

如何获取java.time类?

ThreeTen-Extra项目通过增加类来扩展java.time。该项目是java.time可能未来添加的一个试验场。您可能会在这里找到一些有用的类,例如Interval, YearWeek, YearQuarter更多


3
点赞。更简短的写法:Date now = Date.from(Instant.now().truncatedTo(ChronoUnit.SECONDS));,并且一般情况下不要再使用已经被弃用的 Date 类。 - smirnoff

12

基本答案是,你不能。 Date#toString 返回的值是Date对象的表示形式,除了toString 方法内部使用的格式之外,它没有任何格式概念。

通常情况下,这不应该用于显示目的(除了极少数情况)

相反,您应该使用某种DateFormat

例如...

Date date = new Date();
System.out.println(date);
System.out.println(DateFormat.getDateTimeInstance().format(date));
System.out.println(DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.SHORT).format(date));
System.out.println(DateFormat.getDateTimeInstance(DateFormat.MEDIUM, DateFormat.MEDIUM).format(date));
System.out.println(DateFormat.getDateTimeInstance(DateFormat.LONG, DateFormat.LONG).format(date));

将会输出类似于...

Thu Jan 30 16:29:31 EST 2014
30/01/2014 4:29:31 PM
30/01/14 4:29 PM
30/01/2014 4:29:31 PM
30 January 2014 4:29:31 PM
如果你真的遇到了困难,可以使用SimpleDateFormat进行进一步定制,但如果可能的话,我建议你避免这样做,因为不是每个人都使用相同的日期/时间格式;)

1
这个问题我是在2014年2月回答的,当时Java 8还没有发布,因此我们没有“新而闪亮”的java.timeAPI。那么,基于这个事实,这个答案有什么问题会引起负评?我同意,Basil的答案更加更新,是在这个答案之后3年发表的。 - MadProgrammer

8

将时间精确到秒(去掉毫秒),返回一个新的日期:

public Date truncToSec(Date date) {
    Calendar c = Calendar.getInstance();
    c.setTime(date);
    c.set(Calendar.MILLISECOND, 0);
    Date newDate = c.getTime();
    return newDate;
}

7
您可以使用 SimpleDateFormatter。请参考以下代码。
SimpleDateFormat formatter = new SimpleDateFormat("MM/dd/yyyy hh:mm:ss a");
Date now = date.getTime();
System.out.println(formatter.format(now));

4
使用Apache的DateUtils:
import org.apache.commons.lang.time.DateUtils;
...
DateUtils.truncate(new Date(), Calendar.SECOND)

2
我痛苦地学到,这不会截断毫秒。如果你想要截断毫秒,使用Calendar.SECOND。 - Dargenn
@Dargenn 不好意思,我刚才是在翻译我的代码。我已经修改了我的回答。 - Scott White
还有一个 round 方法,它进行四舍五入而不是截断。适用于那些需要这种功能的人。 - Peterdk

0

仅供参考,您链接的帖子中给出的被接受的答案是可行的:

public static void main(String[] args) {
       SimpleDateFormat df = new SimpleDateFormat("S");
       Date d = new Date();
       System.out.println(df.format(d));
       Calendar c = Calendar.getInstance();
       c.set(Calendar.MILLISECOND, 0);
       d.setTime(c.getTimeInMillis());
       System.out.println(df.format(d));

}

0

-1
请尝试以下日期格式化程序:
import java.text.*;
SimpleDateFormat tmp = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss")
System.out.println(tmp.format(date));

你认为使用 Math.round 函数对一个 long 类型的值进行取整有什么好处呢?它已经是一个整数了... - Jon Skeet
@JonSkeet,谢谢Jon,我已经根据您的反馈更新了答案! - Nick Grealy
你现在拥有与问题中OP相同的代码,但它并不能实现他想要的功能。问题实际上是关于格式的 - 如果OP使用的格式不包括毫秒部分,则根本没有必要首先更改该值。 - Jon Skeet
@JonSkeet,请查看下面的部分,有关格式。 - Nick Grealy
2
“HH:mm:ss SSS” 仍会打印毫秒,而 OP 正试图完全摆脱它们... - MadProgrammer

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接