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

Posted

技术标签:

【中文标题】将时间戳(以毫秒为单位)转换为 Java 中的字符串格式时间【英文标题】:Convert timestamp in milliseconds to string formatted time in Java 【发布时间】:2011-05-07 17:43:24 【问题描述】:

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

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

到目前为止,我已经尝试了以下方法,但失败了:

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)

我该怎么做?

【问题讨论】:

【参考方案1】:

我只想显示时间的相关部分。因此,始终显示秒,但仅显示分钟/小时/天(如果有)。

另外,可选择显示毫秒。

而且我使用的是 GWT,所以我不能使用 String.format。

所以,如果你也是这样,这里是代码。

public static String formatTimeFromMs(long timeInMs, boolean showMs) 
    boolean negative = timeInMs < 0;

    timeInMs = Math.abs(timeInMs);

    StringBuffer result = new StringBuffer();
    long seconds = (timeInMs / 1000) % 60;
    long minutes = (timeInMs / (1000 * 60)) % 60;
    long hours = (timeInMs / (1000 * 60 * 60)) % 24;
    long days = (timeInMs / (1000 * 60 * 60 * 24));

    if (days > 0) 
        result.append(days + "d ");
    
    
    if (hours > 0) 
        if (hours < 10 && result.length() > 0) 
            result.append("0");
        
        result.append(hours + ":");
    
    else if (result.length() > 0) 
        result.append("00:");
    

    if (minutes > 0) 
        if (minutes < 10 && result.length() > 0) 
            result.append("0");
        
        result.append(minutes + ":");
    
    else if (result.length() > 0) 
        result.append("00:");
    

    if (seconds > 0) 
        if (seconds < 10 && result.length() > 0) 
            result.append("0");
        
        result.append(seconds);
    
    else if (result.length() > 0) 
        result.append("00");
    
    else 
        result.append("0");
    

    if (showMs) 
        long millis = timeInMs % 1000;
        
        if (millis < 10) 
            result.append(".00" + millis);
        
        else if (millis < 100) 
            result.append(".0" + millis);
        
        else 
            result.append("." + millis);
        
    

    if (negative) 
        result.insert(0, "-");
    

    return result.toString();

【讨论】:

【参考方案2】:
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);

【讨论】:

【参考方案3】:

试试这个:

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 毫秒的输入查看 runnable example。

【讨论】:

注释:HH 将打印该日期 (0-23) 的小时数,而不是自 1970 年以来经过的总小时数。只是说。 别忘了。旧的 SimpleDateFormat 不能用于多线程。 要导入SimpleDateFormat,使用如下导入:import java.text.*; 仅供参考,麻烦的旧日期时间类,如 java.util.Datejava.util.Calendarjava.text.SimpleDateFormat 现在是 legacy,被 Java 8 及更高版本中内置的 java.time 类所取代.见Tutorial by Oracle。见:***.com/a/4142428/642706 logevent 的类型是什么?可以请您输入问题吗?【参考方案4】:
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);

【讨论】:

我更喜欢您的解决方案,而不是接受的答案,因为它更明确并且不会遇到语言环境问题。虽然你错过了最后一部分:millis = millis % 1000,这会正确地将毫秒放在格式化字符串的末尾。 @WesleyDeKeirsmaeker,总的来说,可读性比简洁更重要。 为什么剩下 24 小时? 除法不相关:50000 / (1000 * 60) = 0.8333333333 而 50000 / 1000 * 60= 3000。 'millis = millis % 1000' 缺失 :D\n 如果你需要几天,你可以去:'long days = (millis /(1000*60*60*24));' 【参考方案5】:

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

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

java.time 框架是使用 Java 8 及更高版本时要走的路。否则,例如 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 );

毫秒SinceEpoch:1289375173771 瞬间:2010-11-10T07:46:13.771Z 输出:07:46:13:771

【讨论】:

是否有任何性能原因或任何其他原因,我应该更喜欢 Joda 而不是 Calendar? 在这种简单的情况下,没有。总的来说:Joda 的 api 设计得更好。例如java 日期是可变的 Joda DateTimes 不是。 Joda 对程序员也更加友好,您可以访问 DateTime 类中的几乎所有功能,而无需在 Date 和 Calendar 之间来回转换 我应该注意它是否有任何依赖关系? 很好的答案,但我建议同时指定时区,而不是隐式依赖 JVM 当前的默认时区。所以对 DateTime 的构造函数的调用会有第二个参数,一个DateTimeZone 对象。像这样:new DateTime( timestamp, DateTimeZone.forID( "America/Montreal" ) ) @Cratylus java.time 类取代了 both Joda-Time 和与 Java 捆绑的旧旧日期时间类。 Joda-Time 启发了 java.time 类,这两个项目由同一个人 Stephen Colbourne 领导。【参考方案6】:

试试这个:

    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);

【讨论】:

【参考方案7】:
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);

【讨论】:

【参考方案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);

【讨论】:

这似乎不正确。例如,TimeUnit.MILLISECONDS.toSeconds() 是否将毫秒转换为相同的持续时间(以秒为单位)或减去小时和分钟后的剩余秒数?解决方案需要后者。 这个方法不正确,如果我想在2019年3月12日的日期,它返回431220:25873204:1552392282:0 >13h04 42s 我下面的回答与此类似,但修复了错误。【参考方案9】:

可以使用 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"

希望对你有帮助...

【讨论】:

【参考方案10】:

在做

logEvent.timeStamp / (1000*60*60)

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

logEvent.timeStamp / (1000*60)

你最终会得到与

相同的答案
TimeUnit.MILLISECONDS.toMinutes(logEvent.timeStamp)

【讨论】:

但是 TimeUnit.MILLISECONDS.toMinutes(logEvent.timeStamp) 也给了我的垃圾。 21489586 不是分钟! 它是自 1970 年 1 月 1 日以来经过的分钟数。 好的,那我怎么得到我想要的格式呢? IE。 10:50:40:450?我不知道如何使用自 1970 年以来的分钟数。 抱歉,您只是想知道如何理解它。请参阅 seanizer 的帖子,应该会涵盖它。

以上是关于将时间戳(以毫秒为单位)转换为 Java 中的字符串格式时间的主要内容,如果未能解决你的问题,请参考以下文章

Java:Unix时间以毫秒为单位

Freemarker 以毫秒为单位将时间戳转换为带有时区的日期

如何将 Mysql 时间戳转换为 sysdate(6) 格式(以毫秒为单位)?

如何编写一个简单的java程序,可以在单个单元中转换一系列时间? (例如以毫秒为单位)[关闭]在线尝试!

Razor 视图中的 Epoch/Unix 时间戳(以毫秒为单位)到日期时间

使用 PHP 将时间戳以毫秒为单位插入 Postgres 表