在Java中将以毫秒为单位的时间戳转换为字符串格式的时间


126

我正在尝试将一个长值(从1970年1月1日经过的毫秒数,即Epoch)转换为格式时间h:m:s:ms

我用作时间戳的长值timestamp,是从log4j的日志记录事件字段中获得的。

到目前为止,我尝试了以下操作,但失败了:

logEvent.timeStamp/ (1000*60*60)
TimeUnit.MILLISECONDS.toMinutes(logEvent.timeStamp)

但是我得到了不正确的值:

1289375173771 for logEvent.timeStamp
358159  for logEvent.timeStamp/ (1000*60*60) 
21489586 for TimeUnit.MILLISECONDS.toMinutes(logEvent.timeStamp)

我该怎么办?

Answers:


188

试试这个:

Date date = new Date(logEvent.timeSTamp);
DateFormat formatter = new SimpleDateFormat("HH:mm:ss.SSS");
formatter.setTimeZone(TimeZone.getTimeZone("UTC"));
String dateFormatted = formatter.format(date);

有关该类接受的其他格式字符串的说明,请参见SimpleDateFormat

请参见使用1200 ms输入的可运行示例


8
注释:HH将打印该日期(0-23)的小时,而不是自1970年以来经过的总小时数。请说。
JohnyTex 2014年

4
而且不要忘记。旧的SimpleDateFormat不能用于多线程。
凯基

要导入SimpleDateFormat,请按以下方式使用导入:import java.text.*;
Naveen Kumar RB

1
仅供参考,麻烦的旧日期,时间类,如java.util.Datejava.util.Calendarjava.text.SimpleDateFormat现在的遗产,由取代java.time内置到Java 8和更高等级。请参见Oracle 教程。请参阅:stackoverflow.com/a/4142428/642706
罗勒·布尔克 Basil Bourque)'18年

1
logevent的类型是什么?您可以输入问题吗?
拉乌普

127
long millis = durationInMillis % 1000;
long second = (durationInMillis / 1000) % 60;
long minute = (durationInMillis / (1000 * 60)) % 60;
long hour = (durationInMillis / (1000 * 60 * 60)) % 24;

String time = String.format("%02d:%02d:%02d.%d", hour, minute, second, millis);

13
我更喜欢您的解决方案,而不是接受的答案,因为它更明确,并且不会遇到语言环境问题。尽管您错过了最后一部分:millis = millis%1000,它将正确地将毫秒数放在格式化字符串的末尾。
2014年

7
@WesleyDeKeirsmaeker,一般而言,可读性比简洁性更重要。
Alphaaa 2014年

2
为什么剩下24小时呢?
baboo 2014年

2
除法不关联:50000 /(1000 * 60)= 0.8333333333,而50000/1000 * 60 =
3000。– farnett

3
'millis = millis%1000'missing:D \ n,如果您需要几天,可以在这里:'long days =(millis /(1000 * 60 * 60 * 24));'
Adreamus 2015年

38

我将向您展示三种方法(a)从长值中获取分钟字段,以及(b)使用所需的日期格式进行打印。一个使用java.util.Calendar,另一个使用Joda-Time,最后一个使用Java 8及更高版本中内置的java.time框架。

java.time框架取代了旧的捆绑日期时间类,并且受Joda-Time(由JSR 310定义,并由ThreeTen-Extra项目扩展)启发。

使用Java 8和更高版本时,可以使用java.time框架。否则,例如Android,请使用Joda-Time。众所周知,java.util.Date/.Calendar类很麻烦,应该避免。

java.util.Date和.Calendar

final long timestamp = new Date().getTime();

// with java.util.Date/Calendar api
final Calendar cal = Calendar.getInstance();
cal.setTimeInMillis(timestamp);
// here's how to get the minutes
final int minutes = cal.get(Calendar.MINUTE);
// and here's how to get the String representation
final String timeString =
    new SimpleDateFormat("HH:mm:ss:SSS").format(cal.getTime());
System.out.println(minutes);
System.out.println(timeString);

乔达时代

// with JodaTime 2.4
final DateTime dt = new DateTime(timestamp);
// here's how to get the minutes
final int minutes2 = dt.getMinuteOfHour();
// and here's how to get the String representation
final String timeString2 = dt.toString("HH:mm:ss:SSS");
System.out.println(minutes2);
System.out.println(timeString2);

输出:

24
09:24:10:254
24
09:24:10:254

java.time

long millisecondsSinceEpoch = 1289375173771L;
Instant instant = Instant.ofEpochMilli ( millisecondsSinceEpoch );
ZonedDateTime zdt = ZonedDateTime.ofInstant ( instant , ZoneOffset.UTC );

DateTimeFormatter formatter = DateTimeFormatter.ofPattern ( "HH:mm:ss:SSS" );
String output = formatter.format ( zdt );

System.out.println ( "millisecondsSinceEpoch: " + millisecondsSinceEpoch + " instant: " + instant + " output: " + output );

msSinceEpoch:1289375173771即时:2010-11-10T07:46:13.771Z输出:07:46:13:771


1
是否有性能原因或其他原因,我应该让乔达(Joda)优先使用Calendar?
Cratylus

1
在像这样的简单情况下,不会。一般而言:Joda的api设计得更好。例如,java日期是可变的,Joda DateTimes不是。Joda对程序员也更友好,您可以访问DateTime类中的几乎所有功能,而无需在Date和Calendar之间来回转换
Sean Patrick Floyd 2010年

它应该有任何依赖项吗?
Cratylus

2
好的答案,但是我建议也指定时区,而不是隐式依赖JVM的当前默认时区。因此,对DateTime构造函数的调用将有第二个参数,即DateTimeZone对象。像这样:new DateTime( timestamp, DateTimeZone.forID( "America/Montreal" ) )
Basil Bourque 2014年

2
@Cratylus java.time类取代 Joda-Time和与Java捆绑在一起的旧的传统日期时间类。Joda-Time启发了java.time类,这两个项目都是由同一个人Stephen Colbourne领导的。
罗勒·布尔克

22

可以使用apache commons(commons-lang3)及其DurationFormatUtils类。

<dependency>
  <groupId>org.apache.commons</groupId>
  <artifactId>commons-lang3</artifactId>
  <version>3.1</version>
</dependency>

例如:

String formattedDuration = DurationFormatUtils.formatDurationHMS(12313152);
// formattedDuration value is "3:25:13.152"
String otherFormattedDuration = DurationFormatUtils.formatDuration(12313152, DurationFormatUtils.ISO_EXTENDED_FORMAT_PATTERN);
// otherFormattedDuration value is "P0000Y0M0DT3H25M13.152S"

希望它可以帮助...


8
long second = TimeUnit.MILLISECONDS.toSeconds(millis);
long minute = TimeUnit.MILLISECONDS.toMinutes(millis);
long hour = TimeUnit.MILLISECONDS.toHours(millis);
millis -= TimeUnit.SECONDS.toMillis(second);
return String.format("%02d:%02d:%02d:%d", hour, minute, second, millis);

2
这似乎不正确。TimeUnit.MILLISECONDS.toSeconds()例如,是否将毫秒转换为相同的持续时间(以秒为单位)或减去小时和分钟后的剩余秒数?解决方案需要后者。
德里克·马哈

此方法不正确,如果我想将2019年
0ddlyoko的

6
public static String timeDifference(long timeDifference1) {
long timeDifference = timeDifference1/1000;
int h = (int) (timeDifference / (3600));
int m = (int) ((timeDifference - (h * 3600)) / 60);
int s = (int) (timeDifference - (h * 3600) - m * 60);

return String.format("%02d:%02d:%02d", h,m,s);

4

在做

logEvent.timeStamp / (1000*60*60)

会给你几个小时,而不是几分钟。尝试:

logEvent.timeStamp / (1000*60)

然后您将得到与以下相同的答案

TimeUnit.MILLISECONDS.toMinutes(logEvent.timeStamp)

但是TimeUnit.MILLISECONDS.toMinutes(logEvent.timeStamp)也给我带来了麻烦。21489586不是分钟!
Cratylus

1
这是自1970年1月1日以来经过的分钟数。
Nico Huysamen 2010年

好的,如何获得所需的格式?即10:50:40:450?我不知道如何使用自1970
。– Cratylus 2010年

抱歉,只是以为您想知道如何理解它。请参阅Seanizer的帖子,该帖子应予以覆盖。
Nico Huysamen 2010年

3

试试这个:

    String sMillis = "10997195233";
    double dMillis = 0;

    int days = 0;
    int hours = 0;
    int minutes = 0;
    int seconds = 0;
    int millis = 0;

    String sTime;

    try {
        dMillis = Double.parseDouble(sMillis);
    } catch (Exception e) {
        System.out.println(e.getMessage());
    }


    seconds = (int)(dMillis / 1000) % 60;
    millis = (int)(dMillis % 1000);

    if (seconds > 0) {
        minutes = (int)(dMillis / 1000 / 60) % 60;
        if (minutes > 0) {
            hours = (int)(dMillis / 1000 / 60 / 60) % 24;
            if (hours > 0) {
                days = (int)(dMillis / 1000 / 60 / 60 / 24);
                if (days > 0) {
                    sTime = days + " days " + hours + " hours " + minutes + " min " + seconds + " sec " + millis + " millisec";
                } else {
                    sTime = hours + " hours " + minutes + " min " + seconds + " sec " + millis + " millisec";
                }
            } else {
                sTime = minutes + " min " + seconds + " sec " + millis + " millisec";
            }
        } else {
            sTime = seconds + " sec " + millis + " millisec";
        }
    } else {
        sTime = dMillis + " millisec";
    }

    System.out.println("time: " + sTime);

0
    long hours = TimeUnit.MILLISECONDS.toHours(timeInMilliseconds);
    long minutes = TimeUnit.MILLISECONDS.toMinutes(timeInMilliseconds - TimeUnit.HOURS.toMillis(hours));
    long seconds = TimeUnit.MILLISECONDS
            .toSeconds(timeInMilliseconds - TimeUnit.HOURS.toMillis(hours) - TimeUnit.MINUTES.toMillis(minutes));
    long milliseconds = timeInMilliseconds - TimeUnit.HOURS.toMillis(hours)
            - TimeUnit.MINUTES.toMillis(minutes) - TimeUnit.SECONDS.toMillis(seconds);
    return String.format("%02d:%02d:%02d:%d", hours, minutes, seconds, milliseconds);
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.