Jak zamienić milisekundy na czytelną dla człowieka formę?

118

Muszę zamienić dowolną liczbę milisekund na dni, godziny, minuty sekundę.

Na przykład: 10 dni, 5 godzin, 13 minut, 1 sekunda.

FlySwat
źródło
„Język, którego używam, nie ma tego wbudowanego, inaczej bym go użył”. Trudno mi to zrozumieć. Jaki język? Jaki system operacyjny?
S.Lott
ActionScript, jakikolwiek system operacyjny, ma
kiepską
3
Nie znam żadnego języka, który ma to, o co prosi, ani nie widzę powodu, dla którego miałby to mieć. Niektóre bardzo proste obliczenia matematyczne na dzielenie / moduły dają dobrą odpowiedź.
Kip
1
Nie wszystkie lata mają taką samą liczbę dni, więc musiałbyś podać, który to był okres. A może po prostu chcesz go mieć w 'standardowych' latach (365. coś)?
Milan Babuškov
@Kip: Rozumiem - źle odczytałem pytanie - myślał o sygnaturach czasowych systemu operacyjnego w milisekundach. Nie czasy delta ani interwały. Kusiło mnie, aby edytować pytanie ...
S.Lott

Odpowiedzi:

226

Cóż, ponieważ nikt inny nie zrobił tego, napiszę prosty kod, aby to zrobić:

x = ms / 1000
seconds = x % 60
x /= 60
minutes = x % 60
x /= 60
hours = x % 24
x /= 24
days = x

Cieszę się, że zatrzymałeś się na kilka dni i nie prosiłeś przez miesiące. :)

Zauważ, że w powyższym założono, że /reprezentuje obcięcie dzielenia liczb całkowitych. Jeśli używasz tego kodu w języku, w którym /reprezentuje dzielenie zmiennoprzecinkowe, będziesz musiał ręcznie obciąć wyniki dzielenia zgodnie z potrzebami.

Greg Hewgill
źródło
2
Po prostu użyłem tego w funkcji flash. dzięki! (za prostotą)
Makram Saleh,
2
Nie działa poprawnie. Powinien używać parseInt, gdy używasz dzielnika, w przeciwnym razie zobaczysz długie wartości zmiennoprzecinkowe. Zobacz moją odpowiedź poniżej, aby uzyskać bardziej kompleksowe rozwiązanie.
Rajiv
17
@Greg Hewgill Cieszę się, że zatrzymałeś się na kilka dni i nie prosiłeś przez miesiące. :) haha :)
moshfiqur
58

Niech A będzie liczbą milisekund. W takim razie masz:

seconds=(A/1000)%60
minutes=(A/(1000*60))%60
hours=(A/(1000*60*60))%24

i tak dalej ( %jest operatorem modułu).

Mam nadzieję że to pomoże.

Gabriele D'Antona
źródło
@sabbibJAVA 24 powinno działać. W jakim jesteś języku Jeśli /dzielenie zmiennoprzecinkowe jest konieczne, musisz skrócić wartość. W innych odpowiedziach zakłada się, że /dokonuje się dzielenia liczb całkowitych.
Brian J
24

Oba poniższe rozwiązania wykorzystują javascript (nie miałem pojęcia, że ​​rozwiązaniem jest agnostyk językowy!). Oba rozwiązania będą musiały zostać przedłużone, jeśli przechwytywanie trwa> 1 month .

Rozwiązanie 1: Użyj obiektu Date

var date = new Date(536643021);
var str = '';
str += date.getUTCDate()-1 + " days, ";
str += date.getUTCHours() + " hours, ";
str += date.getUTCMinutes() + " minutes, ";
str += date.getUTCSeconds() + " seconds, ";
str += date.getUTCMilliseconds() + " millis";
console.log(str);

Daje:

"6 days, 5 hours, 4 minutes, 3 seconds, 21 millis"

Biblioteki są pomocne, ale po co korzystać z biblioteki, skoro można na nowo wynaleźć koło! :)

Rozwiązanie 2: Napisz własny parser

var getDuration = function(millis){
    var dur = {};
    var units = [
        {label:"millis",    mod:1000},
        {label:"seconds",   mod:60},
        {label:"minutes",   mod:60},
        {label:"hours",     mod:24},
        {label:"days",      mod:31}
    ];
    // calculate the individual unit values...
    units.forEach(function(u){
        millis = (millis - (dur[u.label] = (millis % u.mod))) / u.mod;
    });
    // convert object to a string representation...
    var nonZero = function(u){ return dur[u.label]; };
    dur.toString = function(){
        return units
            .reverse()
            .filter(nonZero)
            .map(function(u){
                return dur[u.label] + " " + (dur[u.label]==1?u.label.slice(0,-1):u.label);
            })
            .join(', ');
    };
    return dur;
};

Tworzy obiekt „czasu trwania” z dowolnymi wymaganymi polami. Formatowanie znacznika czasu staje się wtedy proste ...

console.log(getDuration(536643021).toString());

Daje:

"6 days, 5 hours, 4 minutes, 3 seconds, 21 millis"
Nick Grealy
źródło
Zmień tę linię, aby uzyskać liczbę pojedynczą i mnogą return dur[u.label] + " " + (dur[u.label]==1?u.label.slice(0,-1):u.label);
Phillip Kamikaze
1
@PhillipKamikaze Thanks Phillip! Uwzględniłem twoją sugestię.
Nick Grealy
Prawdopodobnie nie chcesz wyświetlać segmentów z zerowymi wartościami, więc można dodać następujący filtr ... var nonZero = function(u){ return !u.startsWith("0"); }; // convert object to a string representation... dur.toString = function(){ return units.reverse().map(function(u){ return dur[u.label] + " " + (dur[u.label]==1?u.label.slice(0,-1):u.label); }).filter(nonZero).join(', '); };
Ruslan Ulanov
1
Dzięki @RuslanUlanov! Dodałem go do przykładu (aczkolwiek z niewielką modyfikacją, aby sprawdzić, czy numer jest „prawdziwy”).
Nick Grealy
7

Powinieneś używać funkcji daty i godziny w dowolnym języku, którego używasz, ale dla zabawy oto kod:

int milliseconds = someNumber;

int seconds = milliseconds / 1000;

int minutes = seconds / 60;

seconds %= 60;

int hours = minutes / 60;

minutes %= 60;

int days = hours / 24;

hours %= 24;
albertein
źródło
4

To jest metoda, którą napisałem. Pobiera integer milliseconds valuei zwraca human-readable String:

public String convertMS(int ms) {
    int seconds = (int) ((ms / 1000) % 60);
    int minutes = (int) (((ms / 1000) / 60) % 60);
    int hours = (int) ((((ms / 1000) / 60) / 60) % 24);

    String sec, min, hrs;
    if(seconds<10)  sec="0"+seconds;
    else            sec= ""+seconds;
    if(minutes<10)  min="0"+minutes;
    else            min= ""+minutes;
    if(hours<10)    hrs="0"+hours;
    else            hrs= ""+hours;

    if(hours == 0)  return min+":"+sec;
    else    return hrs+":"+min+":"+sec;

}
iTurki
źródło
4
function convertTime(time) {        
    var millis= time % 1000;
    time = parseInt(time/1000);
    var seconds = time % 60;
    time = parseInt(time/60);
    var minutes = time % 60;
    time = parseInt(time/60);
    var hours = time % 24;
    var out = "";
    if(hours && hours > 0) out += hours + " " + ((hours == 1)?"hr":"hrs") + " ";
    if(minutes && minutes > 0) out += minutes + " " + ((minutes == 1)?"min":"mins") + " ";
    if(seconds && seconds > 0) out += seconds + " " + ((seconds == 1)?"sec":"secs") + " ";
    if(millis&& millis> 0) out += millis+ " " + ((millis== 1)?"msec":"msecs") + " ";
    return out.trim();
}
Rajiv
źródło
2

Sugerowałbym użycie dowolnych funkcji / bibliotek daty / czasu, które zapewnia wybrany język / framework. Sprawdź również funkcje formatowania ciągów, ponieważ często zapewniają one łatwe sposoby przekazywania daty / znaczników czasu i generowania ciągu w formacie czytelnym dla człowieka.

theraccoonbear
źródło
2

Twoje wybory są proste:

  1. Napisz kod, aby wykonać konwersję (tj. Podziel przez milliSecondsPerDay, aby uzyskać dni, i użyj modułu do podzielenia przez milliSecondsPerHour, aby uzyskać godziny i użyj modułu do podzielenia przez milliSecondsPerMinute i podziel przez 1000 na sekundy. MilliSecondsPerMinute = 60000P milliSeconds milliSecondsPerMinute, milliSecondsPerDay = 24 * milliSecondsPerHour.
  2. Użyj jakiejś procedury operacyjnej. UNIX i Windows mają struktury, które można uzyskać z wartości typu Ticks lub sekund.
cokół
źródło
2
Long serverUptimeSeconds = 
    (System.currentTimeMillis() - SINCE_TIME_IN_MILLISECONDS) / 1000;


String serverUptimeText = 
String.format("%d days %d hours %d minutes %d seconds",
serverUptimeSeconds / 86400,
( serverUptimeSeconds % 86400) / 3600 ,
((serverUptimeSeconds % 86400) % 3600 ) / 60,
((serverUptimeSeconds % 86400) % 3600 ) % 60
);
Krolique
źródło
2
Long expireTime = 69l;
Long tempParam = 0l;

Long seconds = math.mod(expireTime, 60);
tempParam = expireTime - seconds;
expireTime = tempParam/60;
Long minutes = math.mod(expireTime, 60);
tempParam = expireTime - minutes;
expireTime = expireTime/60;
Long hours = math.mod(expireTime, 24);
tempParam = expireTime - hours;
expireTime = expireTime/24;
Long days = math.mod(expireTime, 30);

system.debug(days + '.' + hours + ':' + minutes + ':' + seconds);

Powinno to wydrukować: 0,0: 1: 9

Asit
źródło
2

Dlaczego po prostu nie zrobić czegoś takiego:

var ms = 86400;

var sekund = ms / 1000; //86.4

var minuty = sekundy / 60; //1.4400000000000002

var hours = minuty / 60; //0.024000000000000004

var dni = godziny / 24; //0.0010000000000000002

I radzenie sobie z precyzją zmiennoprzecinkową, np. Number (minutes.toFixed (5)) //1.44

Pavel Blagodov
źródło
2

W java

public static String formatMs(long millis) {
    long hours = TimeUnit.MILLISECONDS.toHours(millis);
    long mins = TimeUnit.MILLISECONDS.toMinutes(millis);
    long secs = TimeUnit.MILLISECONDS.toSeconds(millis);
    return String.format("%dh %d min, %d sec",
            hours,
            mins - TimeUnit.HOURS.toMinutes(hours),
            secs - TimeUnit.MINUTES.toSeconds(mins)
    );
}

Daje coś takiego:

12h 1 min, 34 sec
Camilo Silva
źródło
1

Nie jestem w stanie skomentować pierwszej odpowiedzi na Twoje pytanie, ale jest mały błąd. Powinieneś użyć parseInt lub Math.floor, aby przekonwertować liczby zmiennoprzecinkowe na liczby całkowite, i

var days, hours, minutes, seconds, x;
x = ms / 1000;
seconds = Math.floor(x % 60);
x /= 60;
minutes = Math.floor(x % 60);
x /= 60;
hours = Math.floor(x % 24);
x /= 24;
days = Math.floor(x);

Osobiście używam CoffeeScript w moich projektach i mój kod wygląda tak:

getFormattedTime : (ms)->
        x = ms / 1000
        seconds = Math.floor x % 60
        x /= 60
        minutes = Math.floor x % 60
        x /= 60
        hours = Math.floor x % 24
        x /= 24
        days = Math.floor x
        formattedTime = "#{seconds}s"
        if minutes then formattedTime = "#{minutes}m " + formattedTime
        if hours then formattedTime = "#{hours}h " + formattedTime
        formattedTime 
Rafał Pastuszak
źródło
1

To jest rozwiązanie. Później możesz podzielić przez „:” i pobrać wartości z tablicy

/**
 * Converts milliseconds to human readeable language separated by ":"
 * Example: 190980000 --> 2:05:3 --> 2days 5hours 3min
 */
function dhm(t){
    var cd = 24 * 60 * 60 * 1000,
        ch = 60 * 60 * 1000,
        d = Math.floor(t / cd),
        h = '0' + Math.floor( (t - d * cd) / ch),
        m = '0' + Math.round( (t - d * cd - h * ch) / 60000);
    return [d, h.substr(-2), m.substr(-2)].join(':');
}

var delay = 190980000;                   
var fullTime = dhm(delay);
console.log(fullTime);
ssamuel68
źródło
1

Oto moje rozwiązanie przy użyciu TimeUnit.

UPDATE: Powinienem zaznaczyć, że jest to napisane w stylu groovy, ale Java jest prawie identyczna.

def remainingStr = ""

/* Days */
int days = MILLISECONDS.toDays(remainingTime) as int
remainingStr += (days == 1) ? '1 Day : ' : "${days} Days : "
remainingTime -= DAYS.toMillis(days)

/* Hours */
int hours = MILLISECONDS.toHours(remainingTime) as int
remainingStr += (hours == 1) ? '1 Hour : ' : "${hours} Hours : "
remainingTime -= HOURS.toMillis(hours)

/* Minutes */
int minutes = MILLISECONDS.toMinutes(remainingTime) as int
remainingStr += (minutes == 1) ? '1 Minute : ' : "${minutes} Minutes : "
remainingTime -= MINUTES.toMillis(minutes)

/* Seconds */
int seconds = MILLISECONDS.toSeconds(remainingTime) as int
remainingStr += (seconds == 1) ? '1 Second' : "${seconds} Seconds"
dafunker
źródło
1

Elastyczny sposób na zrobienie tego:
(Nie stworzono na bieżącą datę, ale wystarczająco dobre na okresy)

/**
convert duration to a ms/sec/min/hour/day/week array
@param {int}        msTime              : time in milliseconds 
@param {bool}       fillEmpty(optional) : fill array values even when they are 0.
@param {string[]}   suffixes(optional)  : add suffixes to returned values.
                                        values are filled with missings '0'
@return {int[]/string[]} : time values from higher to lower(ms) range.
*/
var msToTimeList=function(msTime,fillEmpty,suffixes){
    suffixes=(suffixes instanceof Array)?suffixes:[];   //suffixes is optional
    var timeSteps=[1000,60,60,24,7];    // time ranges : ms/sec/min/hour/day/week
    timeSteps.push(1000000);    //add very big time at the end to stop cutting
    var result=[];
    for(var i=0;(msTime>0||i<1||fillEmpty)&&i<timeSteps.length;i++){
        var timerange = msTime%timeSteps[i];
        if(typeof(suffixes[i])=="string"){
            timerange+=suffixes[i]; // add suffix (converting )
            // and fill zeros :
            while(  i<timeSteps.length-1 &&
                    timerange.length<((timeSteps[i]-1)+suffixes[i]).length  )
                timerange="0"+timerange;
        }
        result.unshift(timerange);  // stack time range from higher to lower
        msTime = Math.floor(msTime/timeSteps[i]);
    }
    return result;
};

Uwaga: możesz również ustawić timeSteps jako parametr, jeśli chcesz kontrolować zakresy czasu.

jak używać (skopiuj test):

var elsapsed = Math.floor(Math.random()*3000000000);

console.log(    "elsapsed (labels) = "+
        msToTimeList(elsapsed,false,["ms","sec","min","h","days","weeks"]).join("/")    );

console.log(    "half hour : "+msToTimeList(elsapsed,true)[3]<30?"first":"second"   );

console.log(    "elsapsed (classic) = "+
        msToTimeList(elsapsed,false,["","","","","",""]).join(" : ")    );
yorg
źródło
1

Proponuję skorzystać z http://www.ocpsoft.org/prettytime/ biblioteki.

bardzo łatwo jest uzyskać przedział czasu w postaci czytelnej dla człowieka, na przykład

PrettyTime p = new PrettyTime(); System.out.println(p.format(new Date()));

będzie drukowane jak „chwile od teraz”

inny przykład

PrettyTime p = new PrettyTime()); Date d = new Date(System.currentTimeMillis()); d.setHours(d.getHours() - 1); String ago = p.format(d);

następnie ciąg znaków temu = „1 godzinę temu”

DroidDev
źródło
0

Oto bardziej precyzyjna metoda w JAVA, zaimplementowałem tę prostą logikę, mam nadzieję, że to ci pomoże:

    public String getDuration(String _currentTimemilliSecond)
    {
        long _currentTimeMiles = 1;         
        int x = 0;
        int seconds = 0;
        int minutes = 0;
        int hours = 0;
        int days = 0;
        int month = 0;
        int year = 0;

        try 
        {
            _currentTimeMiles = Long.parseLong(_currentTimemilliSecond);
            /**  x in seconds **/   
            x = (int) (_currentTimeMiles / 1000) ; 
            seconds = x ;

            if(seconds >59)
            {
                minutes = seconds/60 ;

                if(minutes > 59)
                {
                    hours = minutes/60;

                    if(hours > 23)
                    {
                        days = hours/24 ;

                        if(days > 30)
                        {
                            month = days/30;

                            if(month > 11)
                            {
                                year = month/12;

                                Log.d("Year", year);
                                Log.d("Month", month%12);
                                Log.d("Days", days % 30);
                                Log.d("hours ", hours % 24);
                                Log.d("Minutes ", minutes % 60);
                                Log.d("Seconds  ", seconds % 60);   

                                return "Year "+year + " Month "+month%12 +" Days " +days%30 +" hours "+hours%24 +" Minutes "+minutes %60+" Seconds "+seconds%60;
                            }
                            else
                            {
                                Log.d("Month", month);
                                Log.d("Days", days % 30);
                                Log.d("hours ", hours % 24);
                                Log.d("Minutes ", minutes % 60);
                                Log.d("Seconds  ", seconds % 60);   

                                return "Month "+month +" Days " +days%30 +" hours "+hours%24 +" Minutes "+minutes %60+" Seconds "+seconds%60;
                            }

                        }
                        else
                        {
                            Log.d("Days", days );
                            Log.d("hours ", hours % 24);
                            Log.d("Minutes ", minutes % 60);
                            Log.d("Seconds  ", seconds % 60);   

                            return "Days " +days +" hours "+hours%24 +" Minutes "+minutes %60+" Seconds "+seconds%60;
                        }

                    }
                    else
                    {
                        Log.d("hours ", hours);
                        Log.d("Minutes ", minutes % 60);
                        Log.d("Seconds  ", seconds % 60);

                        return "hours "+hours+" Minutes "+minutes %60+" Seconds "+seconds%60;
                    }
                }
                else
                {
                    Log.d("Minutes ", minutes);
                    Log.d("Seconds  ", seconds % 60);

                    return "Minutes "+minutes +" Seconds "+seconds%60;
                }
            }
            else
            {
                Log.d("Seconds ", x);
                return " Seconds "+seconds;
            }
        }
        catch (Exception e) 
        {
            Log.e(getClass().getName().toString(), e.toString());
        }
        return "";
    }

    private Class Log
    {
        public static void d(String tag , int value)
        {
            System.out.println("##### [ Debug ]  ## "+tag +" :: "+value);
        }
    }
AK Joshi
źródło