Перетворіть часову позначку в мілісекундах у форматизований рядок час на Java


126

Я намагаюся перетворити велике значення ( кількість мілісекунд, що минуло з 1.01.1970, тобто епоха ), у формат часу 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)

Як мені це зробити?

Відповіді:


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 мс.


8
Коментар: HH надрукує годину в цю дату (0-23), а не загальну кількість годин, що минула з 1970 року. Просто скажіть.
JohnyTex

4
І не забувайте. Старий SimpleDateFormat не можна використовувати багатопотоково.
keiki

щоб імпортувати SimpleDateFormat, використовуйте імпорт таким чином:import java.text.*;
Naveen Kumar RB

1
FYI, тривожні старі класи дату, такі як java.util.Date, java.util.Calendarі java.text.SimpleDateFormatтепер застарілі , витіснені класами java.time, вбудованими в Java 8 та новіших версій. Дивіться Підручник від Oracle . Дивіться: stackoverflow.com/a/4142428/642706
Василь Бурк

1
Ви можете, будь-ласка, поставити питання?
Раульп

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, яка правильно поставить мілісекунди в кінці форматованого рядка.
Ондрей Буркерт

7
@WesleyDeKeirsmaeker, загалом, читабельність важливіша, ніж стислість.
Альфаа

2
чому залишок з 24 на години?
baboo

2
Відділ не асоціативний: 50000 / (1000 * 60) = 0,8333333333, тоді як 50000/1000 * 60 = 3000.
farnett

3
'millis = millis% 1000' відсутня: D \ n і якщо вам потрібні дні, ви переходите: 'long days = (millis / (1000 * 60 * 60 * 24));'
Адрімус

38

Я покажу вам три способи (а) отримати хвилинне поле з великого значення та (б) роздрукувати його у потрібному форматі дати. Один використовує java.util.Calendar , інший використовує Joda-Time , а останній використовує структуру java.time, вбудовану в Java 8 та новіші версії.

Рамка java.time витісняє старі комплекти дат-часових класів і надихає Joda-Time, визначений JSR 310, і розширений проектом ThreeTen-Extra.

Рамка 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);

Joda-Time

// 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


1
Чи є якась причина роботи чи будь-яка інша причина, я повинен робити перевагу Joda над календарем?
Cratylus

1
у таких простих випадках, як ні. Взагалі: api Joda розроблені краще. Наприклад, дати не змінюються. Joda DateTimes не є. Joda також більш приємний для програмістів, ви можете отримати доступ до майже всіх функціональних можливостей з класу DateTime, не потребуючи перетворення вперед і назад між датою та календарем
Шон Патрік Флойд

чи є у мене залежність, яку я повинен знати?
Cratylus

2
Хороша відповідь, але я пропоную також вказати часовий пояс, а не неявно покладатися на поточний часовий пояс JVM за замовчуванням. Отже, виклик конструктору DateTime мав би другий аргумент - DateTimeZoneоб'єкт. Ось так:new DateTime( timestamp, DateTimeZone.forID( "America/Montreal" ) )
Василь Бурк

2
@Cratylus Класи java.time витісняють як Joda-Time, так і старі застарілі класи дати-часу в комплекті з Java. Joda-Time надихнув заняття java.time, обидва проекти очолював той самий чоловік, Стівен Колбурн.
Василь Бурк

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(), наприклад, конвертувати мілісекунди в той же час в секундах або залишилися секунди після вирахування годин і хвилин? Рішення вимагає останнього.
Дерек Махар

Цей метод невірний, він повертає 431220: 25873204: 1552392282: 0, якщо я хочу перетворити дату 12 березня 2019 о 13:00 04s
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
Це кількість хвилин, що пройшли з 01.01.1970 року.
Ніко Хуйсамен

Гаразд, як я можу отримати потрібний формат? Тобто 10: 50: 40: 450? Я не знаю, як використати кількість хвилин з 1970 року.
Cratylus

О, вибачте, просто думав, що ти хочеш знати, як це зрозуміти. Ознайомтеся із постом моряка, який повинен висвітлювати його.
Ніко Хуйсамен

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);
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.