Come convertire i millisecondi in un formato leggibile dall'uomo?


118

Devo convertire una quantità arbitraria di millisecondi in giorni, ore, minuti secondi.

Ad esempio: 10 giorni, 5 ore, 13 minuti, 1 secondo.


"La lingua che sto usando non lo ha integrato, altrimenti lo userei." Lo trovo difficile da capire. Quale lingua? Quale sistema operativo?
S.Lott

ActionScript, qualsiasi sistema operativo, ha un supporto data / ora miserabile
FlySwat

3
Non conosco nessuna lingua che abbia ciò che sta chiedendo, né vedo alcuna ragione per cui lo farebbe. Alcuni molto semplici calcoli di divisione / modulo danno la risposta giusta.
Kip

1
Non tutti gli anni hanno lo stesso numero di giorni, quindi dovresti indicare quale periodo è stato. O forse lo vuoi solo negli anni "standard" (365, qualcosa)?
Milan Babuškov

@Kip: Capito - ho letto male la domanda - stavo pensando ai timestamp del sistema operativo in millisecondi. Non tempi o intervalli delta. Tentato di modificare la domanda ...
S.Lott

Risposte:


226

Bene, poiché nessun altro si è fatto avanti, scriverò il codice facile per farlo:

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

Sono solo contento che ti sei fermato a giorni e non hai chiesto per mesi. :)

Si noti che in quanto sopra, si presume che /rappresenti il ​​troncamento della divisione intera. Se si utilizza questo codice in una lingua in cui /rappresenta la divisione in virgola mobile, sarà necessario troncare manualmente i risultati della divisione secondo necessità.


2
L'ho appena usato in una funzione flash. Grazie! (votato per semplicità)
Makram Saleh

2
Non funziona correttamente. Dovrebbe usare parseInt quando si usa il divisore altrimenti vedrai valori float lunghi. Vedi la mia risposta di seguito per una soluzione più completa.
Rajiv

17
@ Greg Hewgill Sono solo contento che ti sei fermato a giorni e non hai chiesto per mesi. :) haha :)
moshfiqur

58

Sia A la quantità di millisecondi. Allora hai:

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

e così via ( %è l'operatore modulo).

Spero che questo ti aiuti.


@sabbibJAVA 24 avrebbe dovuto funzionare. In che lingua sei? Se /esegue la divisione in virgola mobile, è necessario troncare il valore. In altre risposte si presume che /stia eseguendo una divisione intera.
Brian J

24

Entrambe le soluzioni di seguito utilizzano javascript (non avevo idea che la soluzione fosse indipendente dalla lingua!). Entrambe le soluzioni dovranno essere estese se si acquisiscono le durate > 1 month.

Soluzione 1: utilizzare l'oggetto Data

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

dà:

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

Le biblioteche sono utili, ma perché usare una libreria quando puoi reinventare la ruota! :)

Soluzione 2: scrivi il tuo 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;
};

Crea un oggetto "durata", con tutti i campi richiesti. La formattazione di un timestamp diventa quindi semplice ...

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

dà:

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

Cambia quella linea per ottenere singolare e plurale return dur[u.label] + " " + (dur[u.label]==1?u.label.slice(0,-1):u.label);
Phillip Kamikaze

1
@PhillipKamikaze Grazie Phillip! Ho incorporato il tuo suggerimento.
Nick Grealy

Probabilmente non vuoi mostrare segmenti con valori zero, quindi il seguente filtro potrebbe essere aggiunto ... 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
Grazie @RuslanUlanov! L'ho aggiunto all'esempio (anche se con una leggera modifica per verificare se il numero è "veritiero").
Nick Grealy


7

Dovresti usare le funzioni datetime di qualunque lingua tu stia utilizzando, ma, solo per divertimento, ecco il codice:

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;

4

Questo è un metodo che ho scritto. Prende un integer milliseconds valuee restituisce un 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;

}

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

2

Suggerirei di utilizzare qualsiasi funzione / libreria di data / ora fornita dal linguaggio / framework di scelta. Controlla anche le funzioni di formattazione delle stringhe poiché spesso forniscono modi semplici per passare data / timestamp e generare un formato di stringa leggibile dall'uomo.


2

Le tue scelte sono semplici:

  1. Scrivi il codice per eseguire la conversione (ad esempio, dividi per milliSecondsPerDay per ottenere giorni e usa il modulo per dividere per milliSecondsPerHour per ottenere ore e usa il modulo per dividere per milliSecondsPerMinute e dividere per 1000 per secondi. MilliSecondsPerMinute = 60000, milliSecondsPerHour = 60 * milliSecondsPerMinute, milliSecondsPerDay = 24 * milliSecondsPerHour.
  2. Usa una routine operativa di qualche tipo. Sia UNIX che Windows hanno strutture che puoi ottenere da un valore di tipo tick o secondi.

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

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

Dovrebbe stampare: 0,0: 1: 9


2

Perché semplicemente non fare qualcosa di simile:

var ms = 86400;

var secondi = ms / 1000; //86.4

var minuti = secondi / 60; //1.4400000000000002

var ore = minuti / 60; //0.024000000000000004

var giorni = ore / 24; //0.0010000000000000002

E si occupa della precisione del float, ad esempio Number (minutes.toFixed (5)) //1.44


2

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

Dà qualcosa del genere:

12h 1 min, 34 sec

1

Non sono in grado di commentare la prima risposta alla tua domanda, ma c'è un piccolo errore. È necessario utilizzare parseInt o Math.floor per convertire i numeri in virgola mobile in interi, 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);

Personalmente, utilizzo CoffeeScript nei miei progetti e il mio codice si presenta così:

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 

1

Questa è una soluzione. Successivamente puoi dividere per ":" e prendere i valori dell'array

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

1

Ecco la mia soluzione usando TimeUnit.

AGGIORNAMENTO: Vorrei sottolineare che questo è scritto in groovy, ma Java è quasi identico.

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"

1

Un modo flessibile per farlo:
(non fatto per la data corrente ma abbastanza buono per le durate)

/**
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;
};

NB: puoi anche impostare timeSteps come parametro se vuoi controllare le fasce orarie.

come usare (copia un 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(" : ")    );

1

Suggerisco di utilizzare http://www.ocpsoft.org/prettytime/ library ..

è molto semplice ottenere l'intervallo di tempo in un formato leggibile dall'uomo come

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

verrà stampato come "momenti da adesso"

altro esempio

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

quindi stringa fa = "1 ora fa"


0

Ecco un metodo più preciso in JAVA, ho implementato questa semplice logica, spero che questo ti aiuti:

    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);
        }
    }
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.