com.feilong.lib.lang3.time.DurationFormatUtils Maven / Gradle / Ivy
Show all versions of feilong Show documentation
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.feilong.lib.lang3.time;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.TimeZone;
import com.feilong.lib.lang3.StringUtils;
import com.feilong.lib.lang3.Validate;
/**
*
* Duration formatting utilities and constants. The following table describes the tokens
* used in the pattern language for formatting.
*
*
* Pattern Tokens
*
* character
* duration element
*
*
* y
* years
*
*
* M
* months
*
*
* d
* days
*
*
* H
* hours
*
*
* m
* minutes
*
*
* s
* seconds
*
*
* S
* milliseconds
*
*
* 'text'
* arbitrary text content
*
*
*
* Note: It's not currently possible to include a single-quote in a format.
*
* Token values are printed using decimal digits.
* A token character can be repeated to ensure that the field occupies a certain minimum
* size. Values will be left-padded with 0 unless padding is disabled in the method invocation.
*
* @since 2.1
*/
public class DurationFormatUtils{
/**
*
* DurationFormatUtils instances should NOT be constructed in standard programming.
*
*
*
* This constructor is public to permit tools that require a JavaBean instance
* to operate.
*
*/
public DurationFormatUtils(){
super();
}
/**
*
* Pattern used with {@code FastDateFormat} and {@code SimpleDateFormat}
* for the ISO 8601 period format used in durations.
*
*
* @see com.feilong.lib.lang3.time.FastDateFormat
* @see java.text.SimpleDateFormat
*/
public static final String ISO_EXTENDED_FORMAT_PATTERN = "'P'yyyy'Y'M'M'd'DT'H'H'm'M's.SSS'S'";
//-----------------------------------------------------------------------
/**
*
* Formats the time gap as a string.
*
*
*
* The format used is ISO 8601-like: {@code HH:mm:ss.SSS}.
*
*
* @param durationMillis
* the duration to format
* @return the formatted duration, not null
* @throws java.lang.IllegalArgumentException
* if durationMillis is negative
*/
public static String formatDurationHMS(final long durationMillis){
return formatDuration(durationMillis, "HH:mm:ss.SSS");
}
/**
*
* Formats the time gap as a string.
*
*
*
* The format used is the ISO 8601 period format.
*
*
*
* This method formats durations using the days and lower fields of the
* ISO format pattern, such as P7D6TH5M4.321S.
*
*
* @param durationMillis
* the duration to format
* @return the formatted duration, not null
* @throws java.lang.IllegalArgumentException
* if durationMillis is negative
*/
public static String formatDurationISO(final long durationMillis){
return formatDuration(durationMillis, ISO_EXTENDED_FORMAT_PATTERN, false);
}
/**
*
* Formats the time gap as a string, using the specified format, and padding with zeros.
*
*
*
* This method formats durations using the days and lower fields of the
* format pattern. Months and larger are not used.
*
*
* @param durationMillis
* the duration to format
* @param format
* the way in which to format the duration, not null
* @return the formatted duration, not null
* @throws java.lang.IllegalArgumentException
* if durationMillis is negative
*/
public static String formatDuration(final long durationMillis,final String format){
return formatDuration(durationMillis, format, true);
}
/**
*
* Formats the time gap as a string, using the specified format.
* Padding the left hand side of numbers with zeroes is optional.
*
*
*
* This method formats durations using the days and lower fields of the
* format pattern. Months and larger are not used.
*
*
* @param durationMillis
* the duration to format
* @param format
* the way in which to format the duration, not null
* @param padWithZeros
* whether to pad the left hand side of numbers with 0's
* @return the formatted duration, not null
* @throws java.lang.IllegalArgumentException
* if durationMillis is negative
*/
public static String formatDuration(final long durationMillis,final String format,final boolean padWithZeros){
inclusiveBetween(0, Long.MAX_VALUE, durationMillis, "durationMillis must not be negative");
final Token[] tokens = lexx(format);
long days = 0;
long hours = 0;
long minutes = 0;
long seconds = 0;
long milliseconds = durationMillis;
if (Token.containsTokenWithValue(tokens, d)){
days = milliseconds / DateUtils.MILLIS_PER_DAY;
milliseconds = milliseconds - (days * DateUtils.MILLIS_PER_DAY);
}
if (Token.containsTokenWithValue(tokens, H)){
hours = milliseconds / DateUtils.MILLIS_PER_HOUR;
milliseconds = milliseconds - (hours * DateUtils.MILLIS_PER_HOUR);
}
if (Token.containsTokenWithValue(tokens, m)){
minutes = milliseconds / DateUtils.MILLIS_PER_MINUTE;
milliseconds = milliseconds - (minutes * DateUtils.MILLIS_PER_MINUTE);
}
if (Token.containsTokenWithValue(tokens, s)){
seconds = milliseconds / DateUtils.MILLIS_PER_SECOND;
milliseconds = milliseconds - (seconds * DateUtils.MILLIS_PER_SECOND);
}
return format(tokens, 0, 0, days, hours, minutes, seconds, milliseconds, padWithZeros);
}
/**
* Validate that the specified primitive value falls between the two
* inclusive values specified; otherwise, throws an exception with the
* specified message.
*
*
* Validate.inclusiveBetween(0, 2, 1, "Not in range");
*
*
* @param start
* the inclusive start value
* @param end
* the inclusive end value
* @param value
* the value to validate
* @param message
* the exception message if invalid, not null
*
* @throws IllegalArgumentException
* if the value falls outside the boundaries
*
* @since 3.3
*/
private static void inclusiveBetween(final long start,final long end,final long value,final String message){
// TODO when breaking BC, consider returning value
if (value < start || value > end){
throw new IllegalArgumentException(message);
}
}
//-----------------------------------------------------------------------
/**
*
* Formats the time gap as a string.
*
*
*
* The format used is the ISO 8601 period format.
*
*
* @param startMillis
* the start of the duration to format
* @param endMillis
* the end of the duration to format
* @return the formatted duration, not null
* @throws java.lang.IllegalArgumentException
* if startMillis is greater than endMillis
*/
public static String formatPeriodISO(final long startMillis,final long endMillis){
return formatPeriod(startMillis, endMillis, ISO_EXTENDED_FORMAT_PATTERN, false, TimeZone.getDefault());
}
/**
*
* Formats the time gap as a string, using the specified format.
* Padding the left hand side of numbers with zeroes is optional.
*
* @param startMillis
* the start of the duration
* @param endMillis
* the end of the duration
* @param format
* the way in which to format the duration, not null
* @return the formatted duration, not null
* @throws java.lang.IllegalArgumentException
* if startMillis is greater than endMillis
*/
public static String formatPeriod(final long startMillis,final long endMillis,final String format){
return formatPeriod(startMillis, endMillis, format, true, TimeZone.getDefault());
}
/**
*
* Formats the time gap as a string, using the specified format.
* Padding the left hand side of numbers with zeroes is optional and
* the timezone may be specified.
*
*
*
* When calculating the difference between months/days, it chooses to
* calculate months first. So when working out the number of months and
* days between January 15th and March 10th, it choose 1 month and
* 23 days gained by choosing January->February = 1 month and then
* calculating days forwards, and not the 1 month and 26 days gained by
* choosing March -> February = 1 month and then calculating days
* backwards.
*
*
*
* For more control, the Joda-Time
* library is recommended.
*
*
* @param startMillis
* the start of the duration
* @param endMillis
* the end of the duration
* @param format
* the way in which to format the duration, not null
* @param padWithZeros
* whether to pad the left hand side of numbers with 0's
* @param timezone
* the millis are defined in
* @return the formatted duration, not null
* @throws java.lang.IllegalArgumentException
* if startMillis is greater than endMillis
*/
public static String formatPeriod(
final long startMillis,
final long endMillis,
final String format,
final boolean padWithZeros,
final TimeZone timezone){
Validate.isTrue(startMillis <= endMillis, "startMillis must not be greater than endMillis");
// Used to optimise for differences under 28 days and
// called formatDuration(millis, format); however this did not work
// over leap years.
// TODO: Compare performance to see if anything was lost by
// losing this optimisation.
final Token[] tokens = lexx(format);
// timezones get funky around 0, so normalizing everything to GMT
// stops the hours being off
final Calendar start = Calendar.getInstance(timezone);
start.setTime(new Date(startMillis));
final Calendar end = Calendar.getInstance(timezone);
end.setTime(new Date(endMillis));
// initial estimates
int milliseconds = end.get(Calendar.MILLISECOND) - start.get(Calendar.MILLISECOND);
int seconds = end.get(Calendar.SECOND) - start.get(Calendar.SECOND);
int minutes = end.get(Calendar.MINUTE) - start.get(Calendar.MINUTE);
int hours = end.get(Calendar.HOUR_OF_DAY) - start.get(Calendar.HOUR_OF_DAY);
int days = end.get(Calendar.DAY_OF_MONTH) - start.get(Calendar.DAY_OF_MONTH);
int months = end.get(Calendar.MONTH) - start.get(Calendar.MONTH);
int years = end.get(Calendar.YEAR) - start.get(Calendar.YEAR);
// each initial estimate is adjusted in case it is under 0
while (milliseconds < 0){
milliseconds += 1000;
seconds -= 1;
}
while (seconds < 0){
seconds += 60;
minutes -= 1;
}
while (minutes < 0){
minutes += 60;
hours -= 1;
}
while (hours < 0){
hours += 24;
days -= 1;
}
if (Token.containsTokenWithValue(tokens, M)){
while (days < 0){
days += start.getActualMaximum(Calendar.DAY_OF_MONTH);
months -= 1;
start.add(Calendar.MONTH, 1);
}
while (months < 0){
months += 12;
years -= 1;
}
if (!Token.containsTokenWithValue(tokens, y) && years != 0){
while (years != 0){
months += 12 * years;
years = 0;
}
}
}else{
// there are no M's in the format string
if (!Token.containsTokenWithValue(tokens, y)){
int target = end.get(Calendar.YEAR);
if (months < 0){
// target is end-year -1
target -= 1;
}
while (start.get(Calendar.YEAR) != target){
days += start.getActualMaximum(Calendar.DAY_OF_YEAR) - start.get(Calendar.DAY_OF_YEAR);
// Not sure I grok why this is needed, but the brutal tests show it is
if (start instanceof GregorianCalendar && start.get(Calendar.MONTH) == Calendar.FEBRUARY
&& start.get(Calendar.DAY_OF_MONTH) == 29){
days += 1;
}
start.add(Calendar.YEAR, 1);
days += start.get(Calendar.DAY_OF_YEAR);
}
years = 0;
}
while (start.get(Calendar.MONTH) != end.get(Calendar.MONTH)){
days += start.getActualMaximum(Calendar.DAY_OF_MONTH);
start.add(Calendar.MONTH, 1);
}
months = 0;
while (days < 0){
days += start.getActualMaximum(Calendar.DAY_OF_MONTH);
months -= 1;
start.add(Calendar.MONTH, 1);
}
}
// The rest of this code adds in values that
// aren't requested. This allows the user to ask for the
// number of months and get the real count and not just 0->11.
if (!Token.containsTokenWithValue(tokens, d)){
hours += 24 * days;
days = 0;
}
if (!Token.containsTokenWithValue(tokens, H)){
minutes += 60 * hours;
hours = 0;
}
if (!Token.containsTokenWithValue(tokens, m)){
seconds += 60 * minutes;
minutes = 0;
}
if (!Token.containsTokenWithValue(tokens, s)){
milliseconds += 1000 * seconds;
seconds = 0;
}
return format(tokens, years, months, days, hours, minutes, seconds, milliseconds, padWithZeros);
}
//-----------------------------------------------------------------------
/**
*
* The internal method to do the formatting.
*
*
* @param tokens
* the tokens
* @param years
* the number of years
* @param months
* the number of months
* @param days
* the number of days
* @param hours
* the number of hours
* @param minutes
* the number of minutes
* @param seconds
* the number of seconds
* @param milliseconds
* the number of millis
* @param padWithZeros
* whether to pad
* @return the formatted string
*/
static String format(
final Token[] tokens,
final long years,
final long months,
final long days,
final long hours,
final long minutes,
final long seconds,
final long milliseconds,
final boolean padWithZeros){
final StringBuilder buffer = new StringBuilder();
boolean lastOutputSeconds = false;
for (final Token token : tokens){
final Object value = token.getValue();
final int count = token.getCount();
if (value instanceof StringBuilder){
buffer.append(value.toString());
}else{
if (value.equals(y)){
buffer.append(paddedValue(years, padWithZeros, count));
lastOutputSeconds = false;
}else if (value.equals(M)){
buffer.append(paddedValue(months, padWithZeros, count));
lastOutputSeconds = false;
}else if (value.equals(d)){
buffer.append(paddedValue(days, padWithZeros, count));
lastOutputSeconds = false;
}else if (value.equals(H)){
buffer.append(paddedValue(hours, padWithZeros, count));
lastOutputSeconds = false;
}else if (value.equals(m)){
buffer.append(paddedValue(minutes, padWithZeros, count));
lastOutputSeconds = false;
}else if (value.equals(s)){
buffer.append(paddedValue(seconds, padWithZeros, count));
lastOutputSeconds = true;
}else if (value.equals(S)){
if (lastOutputSeconds){
// ensure at least 3 digits are displayed even if padding is not selected
final int width = padWithZeros ? Math.max(3, count) : 3;
buffer.append(paddedValue(milliseconds, true, width));
}else{
buffer.append(paddedValue(milliseconds, padWithZeros, count));
}
lastOutputSeconds = false;
}
}
}
return buffer.toString();
}
/**
*
* Converts a {@code long} to a {@code String} with optional
* zero padding.
*
*
* @param value
* the value to convert
* @param padWithZeros
* whether to pad with zeroes
* @param count
* the size to pad to (ignored if {@code padWithZeros} is false)
* @return the string result
*/
private static String paddedValue(final long value,final boolean padWithZeros,final int count){
final String longString = Long.toString(value);
return padWithZeros ? StringUtils.leftPad(longString, count, '0') : longString;
}
static final Object y = "y";
static final Object M = "M";
static final Object d = "d";
static final Object H = "H";
static final Object m = "m";
static final Object s = "s";
static final Object S = "S";
/**
* Parses a classic date format string into Tokens
*
* @param format
* the format to parse, not null
* @return array of Token[]
*/
static Token[] lexx(final String format){
final ArrayList list = new ArrayList<>(format.length());
boolean inLiteral = false;
// Although the buffer is stored in a Token, the Tokens are only
// used internally, so cannot be accessed by other threads
StringBuilder buffer = null;
Token previous = null;
for (int i = 0; i < format.length(); i++){
final char ch = format.charAt(i);
if (inLiteral && ch != '\''){
buffer.append(ch); // buffer can't be null if inLiteral is true
continue;
}
Object value = null;
switch (ch) {
// TODO: Need to handle escaping of '
case '\'':
if (inLiteral){
buffer = null;
inLiteral = false;
}else{
buffer = new StringBuilder();
list.add(new Token(buffer));
inLiteral = true;
}
break;
case 'y':
value = y;
break;
case 'M':
value = M;
break;
case 'd':
value = d;
break;
case 'H':
value = H;
break;
case 'm':
value = m;
break;
case 's':
value = s;
break;
case 'S':
value = S;
break;
default:
if (buffer == null){
buffer = new StringBuilder();
list.add(new Token(buffer));
}
buffer.append(ch);
}
if (value != null){
if (previous != null && previous.getValue().equals(value)){
previous.increment();
}else{
final Token token = new Token(value);
list.add(token);
previous = token;
}
buffer = null;
}
}
if (inLiteral){ // i.e. we have not found the end of the literal
throw new IllegalArgumentException("Unmatched quote in format: " + format);
}
return list.toArray(new Token[0]);
}
//-----------------------------------------------------------------------
/**
* Element that is parsed from the format pattern.
*/
static class Token{
/**
* Helper method to determine if a set of tokens contain a value
*
* @param tokens
* set to look in
* @param value
* to look for
* @return boolean {@code true} if contained
*/
static boolean containsTokenWithValue(final Token[] tokens,final Object value){
for (final Token token : tokens){
if (token.getValue() == value){
return true;
}
}
return false;
}
private final Object value;
private int count;
/**
* Wraps a token around a value. A value would be something like a 'Y'.
*
* @param value
* to wrap
*/
Token(final Object value){
this.value = value;
this.count = 1;
}
/**
* Wraps a token around a repeated number of a value, for example it would
* store 'yyyy' as a value for y and a count of 4.
*
* @param value
* to wrap
* @param count
* to wrap
*/
Token(final Object value, final int count){
this.value = value;
this.count = count;
}
/**
* Adds another one of the value
*/
void increment(){
count++;
}
/**
* Gets the current number of values represented
*
* @return int number of values represented
*/
int getCount(){
return count;
}
/**
* Gets the particular value this token represents.
*
* @return Object value
*/
Object getValue(){
return value;
}
/**
* Supports equality of this Token to another Token.
*
* @param obj2
* Object to consider equality of
* @return boolean {@code true} if equal
*/
@Override
public boolean equals(final Object obj2){
if (obj2 instanceof Token){
final Token tok2 = (Token) obj2;
if (this.value.getClass() != tok2.value.getClass()){
return false;
}
if (this.count != tok2.count){
return false;
}
if (this.value instanceof StringBuilder){
return this.value.toString().equals(tok2.value.toString());
}else if (this.value instanceof Number){
return this.value.equals(tok2.value);
}else{
return this.value == tok2.value;
}
}
return false;
}
/**
* Returns a hash code for the token equal to the
* hash code for the token's value. Thus 'TT' and 'TTTT'
* will have the same hash code.
*
* @return The hash code for the token
*/
@Override
public int hashCode(){
return this.value.hashCode();
}
/**
* Represents this token as a String.
*
* @return String representation of the token
*/
@Override
public String toString(){
return StringUtils.repeat(this.value.toString(), this.count);
}
}
}