将时间戳(以毫秒为单位)转换为 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.Date
、java.util.Calendar
和 java.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 中的字符串格式时间的主要内容,如果未能解决你的问题,请参考以下文章
Freemarker 以毫秒为单位将时间戳转换为带有时区的日期
如何将 Mysql 时间戳转换为 sysdate(6) 格式(以毫秒为单位)?
如何编写一个简单的java程序,可以在单个单元中转换一系列时间? (例如以毫秒为单位)[关闭]在线尝试!