Кто-нибудь знает о библиотеке Java, которая может печатать число в миллисекундах так же, как это делает C #?
Например, длина 123456 мс будет напечатана как 4d1h3m5s.
Кто-нибудь знает о библиотеке Java, которая может печатать число в миллисекундах так же, как это делает C #?
Например, длина 123456 мс будет напечатана как 4d1h3m5s.
%
и, /
чтобы разделить число на части. Почти проще, чем некоторые из предложенных ответов.
/
и %
.
Ответы:
В Joda Time есть довольно хороший способ сделать это с помощью PeriodFormatterBuilder .
Быстрая победа: PeriodFormat.getDefault().print(duration.toPeriod());
например
//import org.joda.time.format.PeriodFormatter;
//import org.joda.time.format.PeriodFormatterBuilder;
//import org.joda.time.Duration;
Duration duration = new Duration(123456); // in milliseconds
PeriodFormatter formatter = new PeriodFormatterBuilder()
.appendDays()
.appendSuffix("d")
.appendHours()
.appendSuffix("h")
.appendMinutes()
.appendSuffix("m")
.appendSeconds()
.appendSuffix("s")
.toFormatter();
String formatted = formatter.print(duration.toPeriod());
System.out.println(formatted);
PeriodType.daysTime()
или .standard()
не помогло
Я построил простое решение, используя Java 8 Duration.toString()
и немного регулярного выражения:
public static String humanReadableFormat(Duration duration) {
return duration.toString()
.substring(2)
.replaceAll("(\\d[HMS])(?!$)", "$1 ")
.toLowerCase();
}
Результат будет выглядеть так:
- 5h
- 7h 15m
- 6h 50m 15s
- 2h 5s
- 0.1s
Если вам не нужны пробелы между ними, просто удалите replaceAll
.
Duration::toString
отформатирован в соответствии с четко определенным и хорошо зарекомендовавшим себя стандартом ISO 8601 .
.replaceAll("\\.\\d+", "")
перед вызовом toLowerCase()
.
Apache commons-lang предоставляет полезный класс, чтобы сделать это, а также DurationFormatUtils
например
DurationFormatUtils.formatDurationHMS( 15362 * 1000 ) )
=> 4: 16: 02.000 (H: m: s.millis)
DurationFormatUtils.formatDurationISO( 15362 * 1000 ) )
=> P0Y0M0DT4H16M2.000S, ср. ISO8601
JodaTime имеет Period
класс, который может представлять такие количества и может быть отображен (через IsoPeriodFormat
) в формате ISO8601 , например PT4D1H3M5S
, например
Period period = new Period(millis);
String formatted = ISOPeriodFormat.standard().print(period);
Если этот формат не тот, который вам нужен, он PeriodFormatterBuilder
позволяет вам собирать произвольные макеты, включая ваш стиль C # 4d1h3m5s
.
new Period(millis).toString()
использует ISOPeriodFormat.standard()
по умолчанию.
В Java 8 вы также можете использовать toString()
метод java.time.Duration
форматирования без внешних библиотек с использованием представления на основе ISO 8601 секунды, такого как PT8H6M12.345S.
Вот как это можно сделать, используя чистый код JDK:
import javax.xml.datatype.DatatypeFactory;
import javax.xml.datatype.Duration;
long diffTime = 215081000L;
Duration duration = DatatypeFactory.newInstance().newDuration(diffTime);
System.out.printf("%02d:%02d:%02d", duration.getDays() * 24 + duration.getHours(), duration.getMinutes(), duration.getSeconds());
org.threeten.extra.AmountFormats.wordBased
Проект ThreeTen-Extra , поддерживаемый Стивеном Колебурном, автором JSR 310, java.time и Joda-Time , имеет AmountFormats
класс, который работает со стандартными классами времени даты Java 8. Тем не менее, он довольно подробный, без возможности для более компактного вывода.
Duration d = Duration.ofMinutes(1).plusSeconds(9).plusMillis(86);
System.out.println(AmountFormats.wordBased(d, Locale.getDefault()));
1 minute, 9 seconds and 86 milliseconds
Java 9+
Duration d1 = Duration.ofDays(0);
d1 = d1.plusHours(47);
d1 = d1.plusMinutes(124);
d1 = d1.plusSeconds(124);
System.out.println(String.format("%s d %sh %sm %ss",
d1.toDaysPart(),
d1.toHoursPart(),
d1.toMinutesPart(),
d1.toSecondsPart()));
2 д 1ч 6м 4с
Альтернативой строительному подходу Joda-Time может быть решение на основе шаблонов . Это предлагает моя библиотека Time4J . Пример использования класса Duration.Formatter (добавлено несколько пробелов для большей читабельности - удаление пробелов даст желаемый стиль C #):
IsoUnit unit = ClockUnit.MILLIS;
Duration<IsoUnit> dur = // normalized duration with multiple components
Duration.of(123456, unit).with(Duration.STD_PERIOD);
Duration.Formatter<IsoUnit> f = // create formatter/parser with optional millis
Duration.Formatter.ofPattern("D'd' h'h' m'm' s[.fff]'s'");
System.out.println(f.format(dur)); // output: 0d 0h 2m 3.456s
Это java.time
средство форматирования также может печатать длительность -API (однако функции нормализации этого типа менее эффективны):
System.out.println(f.format(java.time.Duration.ofMillis(123456))); // output: 0d 0h 2m 3.456s
Ожидание OP о том, что «123456 мс как long будет напечатано как 4d1h3m5s», вычислено явно неверным способом. Я считаю небрежность причиной. Тот же модуль форматирования продолжительности, определенный выше, также может использоваться в качестве анализатора. Следующий код показывает, что «4d1h3m5s» скорее соответствует349385000 = 1000 * (4 * 86400 + 1 * 3600 + 3 * 60 + 5)
:
System.out.println(
f.parse("4d 1h 3m 5s")
.toClockPeriodWithDaysAs24Hours()
.with(unit.only())
.getPartialAmount(unit)); // 349385000
Другой способ - использовать класс net.time4j.PrettyTime
(который также подходит для локализованного вывода и печати относительного времени, такого как «вчера», «следующее воскресенье», «4 дня назад» и т. Д.):
String s = PrettyTime.of(Locale.ENGLISH).print(dur, TextWidth.NARROW);
System.out.println(s); // output: 2m 3s 456ms
s = PrettyTime.of(Locale.ENGLISH).print(dur, TextWidth.WIDE);
System.out.println(s); // output: 2 minutes, 3 seconds, and 456 milliseconds
s = PrettyTime.of(Locale.UK).print(dur, TextWidth.WIDE);
System.out.println(s); // output: 2 minutes, 3 seconds and 456 milliseconds
Ширина текста определяет, используются ли сокращения. Форматом списка также можно управлять, выбирая соответствующий языковой стандарт. Например, в стандартном английском используется запятая Oxform, а в Великобритании - нет. Последняя версия Time4J v5.5 поддерживает более 90 языков и использует переводы на основе CLDR-репозитория (отраслевой стандарт).
Версия Java 8 на основе ответа user678573 :
private static String humanReadableFormat(Duration duration) {
return String.format("%s days and %sh %sm %ss", duration.toDays(),
duration.toHours() - TimeUnit.DAYS.toHours(duration.toDays()),
duration.toMinutes() - TimeUnit.HOURS.toMinutes(duration.toHours()),
duration.getSeconds() - TimeUnit.MINUTES.toSeconds(duration.toMinutes()));
}
... поскольку в Java 8 нет PeriodFormatter и таких методов, как getHours, getMinutes, ...
Я был бы рад увидеть лучшую версию для Java 8.
Я понимаю, что это может не соответствовать вашему варианту использования, но PrettyTime может быть здесь полезен.
PrettyTime p = new PrettyTime();
System.out.println(p.format(new Date()));
//prints: “right now”
System.out.println(p.format(new Date(1000*60*10)));
//prints: “10 minutes from now”
Duration
определен в разумном стандарте ISO 8601 :PnYnMnDTnHnMnS
гдеP
означает «Период» и отмечает начало,T
отделяет часть даты от части времени, а между ними - необязательные вхождения числа с одним -буквенное сокращение. Например,PT4H30M
= четыре с половиной часа.