Konwertuj znacznik czasu w milisekundach na czas sformatowany jako ciąg w języku Java

126

Próbuję przekonwertować długą wartość ( liczbę milisekund, które upłynęły od 1/1/1970 tj. Epoka ) na czas formatu h:m:s:ms.

Długą wartość, której używam jako znacznika czasu, otrzymuję z pola timestampzdarzenia rejestrowania z log4j.

Do tej pory próbowałem następujących rzeczy i kończy się to niepowodzeniem:

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

ale otrzymuję nieprawidłową wartość:

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

Jak mam się do tego zabrać?

Cratylus
źródło

Odpowiedzi:

188

Spróbuj tego:

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

Zobacz SimpleDateFormat, aby uzyskać opis innych ciągów formatu, które akceptuje klasa.

Zobacz działający przykład z wejściem 1200 ms.

manolowar
źródło
8
Komentarz: HH wydrukuje godzinę w tym dniu (0-23), a nie całkowitą liczbę godzin, które upłynęły od 1970 r. Mówię tylko.
JohnyTex,
4
I nie zapomnij. Starej SimpleDateFormat nie można używać w trybie wielowątkowym.
keiki
aby zaimportować SimpleDateFormat, użyj importu w następujący sposób:import java.text.*;
Naveen Kumar RB
1
FYI, kłopotliwe stare klasy daty i godziny, takie jak java.util.Date, java.util.Calendari java.text.SimpleDateFormatsą teraz starsze , zastąpione przez klasy java.time wbudowane w Javę 8 i nowsze. Zobacz samouczek firmy Oracle . Zobacz: stackoverflow.com/a/4142428/642706
Basil Bourque
1
jaki jest typ zdarzenia logicznego? Czy możesz zadać pytanie?
Raulp
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);
Brajendra Pandey
źródło
13
Twoje rozwiązanie podoba mi się bardziej niż zaakceptowana odpowiedź, ponieważ jest nieco bardziej wyraźne i nie zawiera problemów z lokalizacją. Chociaż pomijasz ostatnią część: millis = millis% 1000, co słusznie umieściłoby milisekundy na końcu sformatowanego ciągu.
Ondrej Burkert
7
@WesleyDeKeirsmaeker, ogólnie rzecz biorąc, czytelność jest ważniejsza niż zwięzłość.
Alphaaa
2
dlaczego pozostaje z 24 godzinami?
baboo,
2
Podział nie jest asocjacyjny: 50000 / (1000 * 60) = 0,8333333333 podczas gdy 50000/1000 * 60 = 3000.
farnett
3
Brakuje „millis = millis% 1000”: D \ n, a jeśli potrzebujesz dni, proszę bardzo: „długie dni = (milis / (1000 * 60 * 60 * 24)); '
Adreamus
38

Pokażę ci trzy sposoby (a) uzyskania pola minut z długiej wartości i (b) wydrukowania go w wybranym formacie daty. Jeden używa java.util.Calendar , inny używa Joda-Time , a ostatni używa frameworka java.time wbudowanego w Javę 8 i nowsze.

Struktura java.time zastępuje stare, dołączone klasy data-czas i jest inspirowana przez Joda-Time, zdefiniowaną przez JSR 310 i rozszerzoną przez projekt ThreeTen-Extra.

Framework java.time jest najlepszym rozwiązaniem w przypadku używania języka Java 8 i nowszych. W przeciwnym razie, na przykład w systemie Android, użyj Joda-Time. Klasy java.util.Date/.Calendar są notorycznie kłopotliwe i należy ich unikać.

java.util.Date i .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);

Wynik:

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

milisekundy OdkądEpoch: 1289375173771 natychmiast: 2010-11-10T07: 46: 13,771 Z wyjście: 07: 46: 13: 771

Sean Patrick Floyd
źródło
1
Czy jest jakiś powód wydajności lub inny powód, dla którego powinienem preferować Jodę zamiast Kalendarza?
Cratylus
1
w takich prostych przypadkach nie. Ogólnie: interfejs API Jody jest lepiej zaprojektowany. np. java Daty są zmienne. Joda DateTime nie są. Joda jest również bardziej przyjazny dla programistów, możesz uzyskać dostęp do prawie wszystkich funkcji z klasy DateTime bez konieczności przełączania między datą a kalendarzem
Sean Patrick Floyd
czy ma jakieś zależności, o których powinienem wiedzieć?
Cratylus
2
Dobra odpowiedź, ale sugeruję również określenie strefy czasowej zamiast polegać niejawnie na bieżącej domyślnej strefie czasowej maszyny JVM. Więc wywołanie konstruktora DateTime miałoby drugi argument, DateTimeZoneobiekt. W ten sposób:new DateTime( timestamp, DateTimeZone.forID( "America/Montreal" ) )
Basil Bourque
2
@Cratylus Klasy java.time zastępują zarówno Joda-Time, jak i starsze starsze klasy daty i czasu dołączone do języka Java. Joda-Time zainspirowała zajęcia java.time, oba projekty prowadzone przez tego samego człowieka, Stephena Colbourne'a.
Basil Bourque
22

Możliwe jest użycie apache commons (commons-lang3) i jego klasy DurationFormatUtils.

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

Na przykład:

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"

Mam nadzieję, że to pomoże ...

YoCha
źródło
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);
Lo Ka
źródło
2
To nie wydaje się prawidłowe. Czy TimeUnit.MILLISECONDS.toSeconds()na przykład konwertuje milisekundy na ten sam czas trwania w sekundach lub pozostałe sekundy po odjęciu godzin i minut? Rozwiązanie wymaga tego drugiego.
Derek Mahar,
Ta metoda jest niepoprawna, zwraca 431220: 25873204: 1552392282: 0, jeśli chcę przekonwertować datę 12 marca 2019 na 13h04 42s
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);
Vivek Pal
źródło
4

Robić

logEvent.timeStamp / (1000*60*60)

da ci godziny, a nie minuty. Próbować:

logEvent.timeStamp / (1000*60)

a otrzymasz taką samą odpowiedź jak

TimeUnit.MILLISECONDS.toMinutes(logEvent.timeStamp)
Nico Huysamen
źródło
Ale TimeUnit.MILLISECONDS.toMinutes (logEvent.timeStamp) również daje moje śmieci. 21489586 to nie minuty!
Cratylus
1
Jest to liczba minut, które minęły od 01.01.1970.
Nico Huysamen
Ok, więc jak uzyskać żądany format? To znaczy 10: 50: 40: 450? Nie wiem, jak wykorzystać liczbę minut od 1970 r.
Cratylus
Przepraszam, po prostu pomyślałem, że chcesz wiedzieć, jak to zrozumieć. Zobacz post seanizera, który powinien to obejmować.
Nico Huysamen
3

Spróbuj tego:

    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);
LeoZao
źródło
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);
Joel Richard Koett
źródło