【问题标题】:Convert timestamp in milliseconds to string formatted time in Java将时间戳(以毫秒为单位)转换为 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)

我该怎么做?

【问题讨论】:

    标签: java datetime time timestamp epoch


    【解决方案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。见:stackoverflow.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 的帖子,应该会涵盖它。
              猜你喜欢
              • 1970-01-01
              • 2018-05-04
              • 2015-06-17
              • 1970-01-01
              • 2016-08-01
              • 2014-12-08
              • 2017-05-28
              • 2014-10-16
              • 1970-01-01
              相关资源
              最近更新 更多