DateFormatUtils.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3.time;

  18. import java.util.Calendar;
  19. import java.util.Date;
  20. import java.util.Locale;
  21. import java.util.TimeZone;

  22. /**
  23.  * Date and time formatting utilities and constants.
  24.  *
  25.  * <p>Formatting is performed using the thread-safe
  26.  * {@link org.apache.commons.lang3.time.FastDateFormat} class.</p>
  27.  *
  28.  * <p>Note that the JDK has a bug wherein calling Calendar.get(int) will
  29.  * override any previously called Calendar.clear() calls. See LANG-755.</p>
  30.  *
  31.  * <p>Note that when using capital YYYY instead of lowercase yyyy, the formatter
  32.  * will assume current year as week year is not supported. See {@link java.util.GregorianCalendar}
  33.  * Week Year section for an explanation on the difference between calendar and week years.</p>
  34.  *
  35.  * @since 2.0
  36.  */
  37. public class DateFormatUtils {

  38.     /**
  39.      * The UTC time zone (often referred to as GMT).
  40.      * This is private as it is mutable.
  41.      */
  42.     private static final TimeZone UTC_TIME_ZONE = FastTimeZone.getGmtTimeZone();

  43.     /**
  44.      * ISO 8601 formatter for date-time without time zone.
  45.      *
  46.      * <p>
  47.      * The format used is {@code yyyy-MM-dd'T'HH:mm:ss}. This format uses the
  48.      * default TimeZone in effect at the time of loading DateFormatUtils class.
  49.      * </p>
  50.      *
  51.      * @since 3.5
  52.      */
  53.     public static final FastDateFormat ISO_8601_EXTENDED_DATETIME_FORMAT
  54.             = FastDateFormat.getInstance("yyyy-MM-dd'T'HH:mm:ss");

  55.     /**
  56.      * @deprecated - as of 4.0, ISO_DATETIME_FORMAT will be replaced by ISO_8601_EXTENDED_DATETIME_FORMAT.
  57.      */
  58.     @Deprecated
  59.     public static final FastDateFormat ISO_DATETIME_FORMAT = ISO_8601_EXTENDED_DATETIME_FORMAT;

  60.     /**
  61.      * ISO 8601 formatter for date-time with time zone.
  62.      *
  63.      * <p>
  64.      * The format used is {@code yyyy-MM-dd'T'HH:mm:ssZZ}. This format uses the
  65.      * default TimeZone in effect at the time of loading DateFormatUtils class.
  66.      * </p>
  67.      *
  68.      * @since 3.5
  69.      */
  70.     public static final FastDateFormat ISO_8601_EXTENDED_DATETIME_TIME_ZONE_FORMAT
  71.             = FastDateFormat.getInstance("yyyy-MM-dd'T'HH:mm:ssZZ");

  72.     /**
  73.      * @deprecated - as of 4.0, ISO_DATETIME_TIME_ZONE_FORMAT will be replaced by ISO_8601_EXTENDED_DATETIME_TIME_ZONE_FORMAT.
  74.      */
  75.     @Deprecated
  76.     public static final FastDateFormat ISO_DATETIME_TIME_ZONE_FORMAT = ISO_8601_EXTENDED_DATETIME_TIME_ZONE_FORMAT;

  77.     /**
  78.      * ISO 8601 formatter for date without time zone.
  79.      *
  80.      * <p>
  81.      * The format used is {@code yyyy-MM-dd}. This format uses the
  82.      * default TimeZone in effect at the time of loading DateFormatUtils class.
  83.      * </p>
  84.      *
  85.      * @since 3.5
  86.      */
  87.     public static final FastDateFormat ISO_8601_EXTENDED_DATE_FORMAT
  88.             = FastDateFormat.getInstance("yyyy-MM-dd");

  89.     /**
  90.      * @deprecated - as of 4.0, ISO_DATE_FORMAT will be replaced by ISO_8601_EXTENDED_DATE_FORMAT.
  91.      */
  92.     @Deprecated
  93.     public static final FastDateFormat ISO_DATE_FORMAT = ISO_8601_EXTENDED_DATE_FORMAT;

  94.     /**
  95.      * ISO 8601-like formatter for date with time zone.
  96.      *
  97.      * <p>
  98.      * The format used is {@code yyyy-MM-ddZZ}. This pattern does not comply
  99.      * with the formal ISO 8601 specification as the standard does not allow
  100.      * a time zone  without a time. This format uses the default TimeZone in
  101.      * effect at the time of loading DateFormatUtils class.
  102.      * </p>
  103.      *
  104.      * @deprecated - as of 4.0, ISO_DATE_TIME_ZONE_FORMAT will be removed.
  105.      */
  106.     @Deprecated
  107.     public static final FastDateFormat ISO_DATE_TIME_ZONE_FORMAT
  108.             = FastDateFormat.getInstance("yyyy-MM-ddZZ");

  109.     /**
  110.      * Non-compliant formatter for time without time zone (ISO 8601 does not
  111.      * prefix 'T' for standalone time value).
  112.      *
  113.      * <p>
  114.      * The format used is {@code 'T'HH:mm:ss}. This format uses the default
  115.      * TimeZone in effect at the time of loading DateFormatUtils class.
  116.      * </p>
  117.      *
  118.      * @deprecated - as of 4.0, ISO_TIME_FORMAT will be removed.
  119.      */
  120.     @Deprecated
  121.     public static final FastDateFormat ISO_TIME_FORMAT
  122.             = FastDateFormat.getInstance("'T'HH:mm:ss");

  123.     /**
  124.      * Non-compliant formatter for time with time zone (ISO 8601 does not
  125.      * prefix 'T' for standalone time value).
  126.      *
  127.      * <p>
  128.      * The format used is {@code 'T'HH:mm:ssZZ}. This format uses the default
  129.      * TimeZone in effect at the time of loading DateFormatUtils class.
  130.      * </p>
  131.      *
  132.      * @deprecated - as of 4.0, ISO_TIME_TIME_ZONE_FORMAT will be removed.
  133.      */
  134.     @Deprecated
  135.     public static final FastDateFormat ISO_TIME_TIME_ZONE_FORMAT
  136.             = FastDateFormat.getInstance("'T'HH:mm:ssZZ");

  137.     /**
  138.      * ISO 8601 formatter for time without time zone.
  139.      *
  140.      * <p>
  141.      * The format used is {@code HH:mm:ss}. This format uses the default
  142.      * TimeZone in effect at the time of loading DateFormatUtils class.
  143.      * </p>
  144.      *
  145.      * @since 3.5
  146.      */
  147.     public static final FastDateFormat ISO_8601_EXTENDED_TIME_FORMAT
  148.             = FastDateFormat.getInstance("HH:mm:ss");

  149.     /**
  150.      * @deprecated - as of 4.0, ISO_TIME_NO_T_FORMAT will be replaced by ISO_8601_EXTENDED_TIME_FORMAT.
  151.      */
  152.     @Deprecated
  153.     public static final FastDateFormat ISO_TIME_NO_T_FORMAT = ISO_8601_EXTENDED_TIME_FORMAT;

  154.     /**
  155.      * ISO 8601 formatter for time with time zone.
  156.      *
  157.      * <p>
  158.      * The format used is {@code HH:mm:ssZZ}. This format uses the default
  159.      * TimeZone in effect at the time of loading DateFormatUtils class.
  160.      * </p>
  161.      *
  162.      * @since 3.5
  163.      */
  164.     public static final FastDateFormat ISO_8601_EXTENDED_TIME_TIME_ZONE_FORMAT
  165.             = FastDateFormat.getInstance("HH:mm:ssZZ");

  166.     /**
  167.      * @deprecated - as of 4.0, ISO_TIME_NO_T_TIME_ZONE_FORMAT will be replaced by ISO_8601_EXTENDED_TIME_TIME_ZONE_FORMAT.
  168.      */
  169.     @Deprecated
  170.     public static final FastDateFormat ISO_TIME_NO_T_TIME_ZONE_FORMAT = ISO_8601_EXTENDED_TIME_TIME_ZONE_FORMAT;

  171.     /**
  172.      * SMTP (and probably other) date headers.
  173.      *
  174.      * <p>
  175.      * The format used is {@code EEE, dd MMM yyyy HH:mm:ss Z} in US locale.
  176.      * This format uses the default TimeZone in effect at the time of loading
  177.      * DateFormatUtils class.
  178.      * </p>
  179.      */
  180.     public static final FastDateFormat SMTP_DATETIME_FORMAT
  181.             = FastDateFormat.getInstance("EEE, dd MMM yyyy HH:mm:ss Z", Locale.US);

  182.     /**
  183.      * Formats a calendar into a specific pattern. The TimeZone from the calendar
  184.      * will be used for formatting.
  185.      *
  186.      * @param calendar  the calendar to format, not null
  187.      * @param pattern  the pattern to use to format the calendar, not null
  188.      * @return the formatted calendar
  189.      * @see FastDateFormat#format(Calendar)
  190.      * @since 2.4
  191.      */
  192.     public static String format(final Calendar calendar, final String pattern) {
  193.         return format(calendar, pattern, getTimeZone(calendar), null);
  194.     }

  195.     /**
  196.      * Formats a calendar into a specific pattern in a locale. The TimeZone from the calendar
  197.      * will be used for formatting.
  198.      *
  199.      * @param calendar  the calendar to format, not null
  200.      * @param pattern  the pattern to use to format the calendar, not null
  201.      * @param locale  the locale to use, may be {@code null}
  202.      * @return the formatted calendar
  203.      * @see FastDateFormat#format(Calendar)
  204.      * @since 2.4
  205.      */
  206.     public static String format(final Calendar calendar, final String pattern, final Locale locale) {
  207.         return format(calendar, pattern, getTimeZone(calendar), locale);
  208.     }

  209.     /**
  210.      * Formats a calendar into a specific pattern in a time zone.
  211.      *
  212.      * @param calendar  the calendar to format, not null
  213.      * @param pattern  the pattern to use to format the calendar, not null
  214.      * @param timeZone  the time zone  to use, may be {@code null}
  215.      * @return the formatted calendar
  216.      * @see FastDateFormat#format(Calendar)
  217.      * @since 2.4
  218.      */
  219.     public static String format(final Calendar calendar, final String pattern, final TimeZone timeZone) {
  220.         return format(calendar, pattern, timeZone, null);
  221.     }

  222.     /**
  223.      * Formats a calendar into a specific pattern in a time zone and locale.
  224.      *
  225.      * @param calendar  the calendar to format, not null
  226.      * @param pattern  the pattern to use to format the calendar, not null
  227.      * @param timeZone  the time zone  to use, may be {@code null}
  228.      * @param locale  the locale to use, may be {@code null}
  229.      * @return the formatted calendar
  230.      * @see FastDateFormat#format(Calendar)
  231.      * @since 2.4
  232.      */
  233.     public static String format(final Calendar calendar, final String pattern, final TimeZone timeZone, final Locale locale) {
  234.         final FastDateFormat df = FastDateFormat.getInstance(pattern, timeZone, locale);
  235.         return df.format(calendar);
  236.     }

  237.     /**
  238.      * Formats a date/time into a specific pattern.
  239.      *
  240.      * @param date  the date to format, not null
  241.      * @param pattern  the pattern to use to format the date, not null
  242.      * @return the formatted date
  243.      */
  244.     public static String format(final Date date, final String pattern) {
  245.         return format(date, pattern, null, null);
  246.     }

  247.     /**
  248.      * Formats a date/time into a specific pattern in a locale.
  249.      *
  250.      * @param date  the date to format, not null
  251.      * @param pattern  the pattern to use to format the date, not null
  252.      * @param locale  the locale to use, may be {@code null}
  253.      * @return the formatted date
  254.      */
  255.     public static String format(final Date date, final String pattern, final Locale locale) {
  256.         return format(date, pattern, null, locale);
  257.     }

  258.     /**
  259.      * Formats a date/time into a specific pattern in a time zone.
  260.      *
  261.      * @param date  the date to format, not null
  262.      * @param pattern  the pattern to use to format the date, not null
  263.      * @param timeZone  the time zone  to use, may be {@code null}
  264.      * @return the formatted date
  265.      */
  266.     public static String format(final Date date, final String pattern, final TimeZone timeZone) {
  267.         return format(date, pattern, timeZone, null);
  268.     }

  269.     /**
  270.      * Formats a date/time into a specific pattern in a time zone and locale.
  271.      *
  272.      * @param date  the date to format, not null
  273.      * @param pattern  the pattern to use to format the date, not null, not null
  274.      * @param timeZone  the time zone  to use, may be {@code null}
  275.      * @param locale  the locale to use, may be {@code null}
  276.      * @return the formatted date
  277.      */
  278.     public static String format(final Date date, final String pattern, final TimeZone timeZone, final Locale locale) {
  279.         final FastDateFormat df = FastDateFormat.getInstance(pattern, timeZone, locale);
  280.         return df.format(date);
  281.     }

  282.     /**
  283.      * Formats a date/time into a specific pattern.
  284.      *
  285.      * @param millis  the date to format expressed in milliseconds
  286.      * @param pattern  the pattern to use to format the date, not null
  287.      * @return the formatted date
  288.      */
  289.     public static String format(final long millis, final String pattern) {
  290.         return format(new Date(millis), pattern, null, null);
  291.     }

  292.     /**
  293.      * Formats a date/time into a specific pattern in a locale.
  294.      *
  295.      * @param millis  the date to format expressed in milliseconds
  296.      * @param pattern  the pattern to use to format the date, not null
  297.      * @param locale  the locale to use, may be {@code null}
  298.      * @return the formatted date
  299.      */
  300.     public static String format(final long millis, final String pattern, final Locale locale) {
  301.         return format(new Date(millis), pattern, null, locale);
  302.     }

  303.     /**
  304.      * Formats a date/time into a specific pattern in a time zone.
  305.      *
  306.      * @param millis  the time expressed in milliseconds
  307.      * @param pattern  the pattern to use to format the date, not null
  308.      * @param timeZone  the time zone  to use, may be {@code null}
  309.      * @return the formatted date
  310.      */
  311.     public static String format(final long millis, final String pattern, final TimeZone timeZone) {
  312.         return format(new Date(millis), pattern, timeZone, null);
  313.     }

  314.     /**
  315.      * Formats a date/time into a specific pattern in a time zone and locale.
  316.      *
  317.      * @param millis  the date to format expressed in milliseconds
  318.      * @param pattern  the pattern to use to format the date, not null
  319.      * @param timeZone  the time zone  to use, may be {@code null}
  320.      * @param locale  the locale to use, may be {@code null}
  321.      * @return the formatted date
  322.      */
  323.     public static String format(final long millis, final String pattern, final TimeZone timeZone, final Locale locale) {
  324.         return format(new Date(millis), pattern, timeZone, locale);
  325.     }

  326.     /**
  327.      * Formats a date/time into a specific pattern using the UTC time zone.
  328.      *
  329.      * @param date  the date to format, not null
  330.      * @param pattern  the pattern to use to format the date, not null
  331.      * @return the formatted date
  332.      */
  333.     public static String formatUTC(final Date date, final String pattern) {
  334.         return format(date, pattern, UTC_TIME_ZONE, null);
  335.     }

  336.     /**
  337.      * Formats a date/time into a specific pattern using the UTC time zone.
  338.      *
  339.      * @param date  the date to format, not null
  340.      * @param pattern  the pattern to use to format the date, not null
  341.      * @param locale  the locale to use, may be {@code null}
  342.      * @return the formatted date
  343.      */
  344.     public static String formatUTC(final Date date, final String pattern, final Locale locale) {
  345.         return format(date, pattern, UTC_TIME_ZONE, locale);
  346.     }

  347.     /**
  348.      * Formats a date/time into a specific pattern using the UTC time zone.
  349.      *
  350.      * @param millis  the date to format expressed in milliseconds
  351.      * @param pattern  the pattern to use to format the date, not null
  352.      * @return the formatted date
  353.      */
  354.     public static String formatUTC(final long millis, final String pattern) {
  355.         return format(new Date(millis), pattern, UTC_TIME_ZONE, null);
  356.     }

  357.     /**
  358.      * Formats a date/time into a specific pattern using the UTC time zone.
  359.      *
  360.      * @param millis  the date to format expressed in milliseconds
  361.      * @param pattern  the pattern to use to format the date, not null
  362.      * @param locale  the locale to use, may be {@code null}
  363.      * @return the formatted date
  364.      */
  365.     public static String formatUTC(final long millis, final String pattern, final Locale locale) {
  366.         return format(new Date(millis), pattern, UTC_TIME_ZONE, locale);
  367.     }

  368.     private static TimeZone getTimeZone(final Calendar calendar) {
  369.         return calendar == null ? null : calendar.getTimeZone();
  370.     }

  371.     /**
  372.      * DateFormatUtils instances should NOT be constructed in standard programming.
  373.      *
  374.      * <p>This constructor is public to permit tools that require a JavaBean instance
  375.      * to operate.</p>
  376.      *
  377.      * @deprecated TODO Make private in 4.0.
  378.      */
  379.     @Deprecated
  380.     public DateFormatUtils() {
  381.         // empty
  382.     }

  383. }