AbstractFormatCache.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.text.DateFormat;
  19. import java.text.Format;
  20. import java.text.SimpleDateFormat;
  21. import java.util.Arrays;
  22. import java.util.Locale;
  23. import java.util.Objects;
  24. import java.util.TimeZone;
  25. import java.util.concurrent.ConcurrentHashMap;
  26. import java.util.concurrent.ConcurrentMap;

  27. import org.apache.commons.lang3.LocaleUtils;

  28. /**
  29.  * FormatCache is a cache and factory for {@link Format}s.
  30.  *
  31.  * @param <F> The Format type.
  32.  *
  33.  * @since 3.0
  34.  */
  35. // TODO: Before making public move from getDateTimeInstance(Integer, ...) to int; or some other approach.
  36. abstract class AbstractFormatCache<F extends Format> {

  37.     /**
  38.      * Helper class to hold multipart Map keys as arrays.
  39.      */
  40.     private static final class ArrayKey {

  41.         private static int computeHashCode(final Object[] keys) {
  42.             final int prime = 31;
  43.             int result = 1;
  44.             result = prime * result + Arrays.hashCode(keys);
  45.             return result;
  46.         }

  47.         private final Object[] keys;
  48.         private final int hashCode;

  49.         /**
  50.          * Constructs an instance of {@link MultipartKey} to hold the specified objects.
  51.          *
  52.          * @param keys the set of objects that make up the key.  Each key may be null.
  53.          */
  54.         ArrayKey(final Object... keys) {
  55.             this.keys = keys;
  56.             this.hashCode = computeHashCode(keys);
  57.         }

  58.         @Override
  59.         public boolean equals(final Object obj) {
  60.             if (this == obj) {
  61.                 return true;
  62.             }
  63.             if (obj == null) {
  64.                 return false;
  65.             }
  66.             if (getClass() != obj.getClass()) {
  67.                 return false;
  68.             }
  69.             final ArrayKey other = (ArrayKey) obj;
  70.             return Arrays.deepEquals(keys, other.keys);
  71.         }

  72.         @Override
  73.         public int hashCode() {
  74.             return hashCode;
  75.         }

  76.     }

  77.     /**
  78.      * No date or no time.  Used in same parameters as DateFormat.SHORT or DateFormat.LONG
  79.      */
  80.     static final int NONE = -1;

  81.     private static final ConcurrentMap<ArrayKey, String> cDateTimeInstanceCache = new ConcurrentHashMap<>(7);

  82.     /**
  83.      * Gets a date/time format for the specified styles and locale.
  84.      *
  85.      * @param dateStyle  date style: FULL, LONG, MEDIUM, or SHORT, null indicates no date in format
  86.      * @param timeStyle  time style: FULL, LONG, MEDIUM, or SHORT, null indicates no time in format
  87.      * @param locale  The non-null locale of the desired format
  88.      * @return a localized standard date/time format
  89.      * @throws IllegalArgumentException if the Locale has no date/time pattern defined
  90.      */
  91.     // package protected, for access from test code; do not make public or protected
  92.     static String getPatternForStyle(final Integer dateStyle, final Integer timeStyle, final Locale locale) {
  93.         final Locale safeLocale = LocaleUtils.toLocale(locale);
  94.         final ArrayKey key = new ArrayKey(dateStyle, timeStyle, safeLocale);
  95.         return cDateTimeInstanceCache.computeIfAbsent(key, k -> {
  96.             try {
  97.                 final DateFormat formatter;
  98.                 if (dateStyle == null) {
  99.                     formatter = DateFormat.getTimeInstance(timeStyle.intValue(), safeLocale);
  100.                 } else if (timeStyle == null) {
  101.                     formatter = DateFormat.getDateInstance(dateStyle.intValue(), safeLocale);
  102.                 } else {
  103.                     formatter = DateFormat.getDateTimeInstance(dateStyle.intValue(), timeStyle.intValue(), safeLocale);
  104.                 }
  105.                 return ((SimpleDateFormat) formatter).toPattern();
  106.             } catch (final ClassCastException ex) {
  107.                 throw new IllegalArgumentException("No date time pattern for locale: " + safeLocale);
  108.             }
  109.         });
  110.     }

  111.     private final ConcurrentMap<ArrayKey, F> cInstanceCache = new ConcurrentHashMap<>(7);

  112.     /**
  113.      * Create a format instance using the specified pattern, time zone
  114.      * and locale.
  115.      *
  116.      * @param pattern  {@link java.text.SimpleDateFormat} compatible pattern, this will not be null.
  117.      * @param timeZone  time zone, this will not be null.
  118.      * @param locale  locale, this will not be null.
  119.      * @return a pattern based date/time formatter
  120.      * @throws IllegalArgumentException if pattern is invalid
  121.      *  or {@code null}
  122.      */
  123.     protected abstract F createInstance(String pattern, TimeZone timeZone, Locale locale);

  124.     /**
  125.      * Gets a date formatter instance using the specified style,
  126.      * time zone and locale.
  127.      *
  128.      * @param dateStyle  date style: FULL, LONG, MEDIUM, or SHORT
  129.      * @param timeZone  optional time zone, overrides time zone of
  130.      *  formatted date, null means use default Locale
  131.      * @param locale  optional locale, overrides system locale
  132.      * @return a localized standard date/time formatter
  133.      * @throws IllegalArgumentException if the Locale has no date/time
  134.      *  pattern defined
  135.      */
  136.     // package protected, for access from FastDateFormat; do not make public or protected
  137.     F getDateInstance(final int dateStyle, final TimeZone timeZone, final Locale locale) {
  138.         return getDateTimeInstance(Integer.valueOf(dateStyle), null, timeZone, locale);
  139.     }

  140.     /**
  141.      * Gets a date/time formatter instance using the specified style,
  142.      * time zone and locale.
  143.      *
  144.      * @param dateStyle  date style: FULL, LONG, MEDIUM, or SHORT
  145.      * @param timeStyle  time style: FULL, LONG, MEDIUM, or SHORT
  146.      * @param timeZone  optional time zone, overrides time zone of
  147.      *  formatted date, null means use default Locale
  148.      * @param locale  optional locale, overrides system locale
  149.      * @return a localized standard date/time formatter
  150.      * @throws IllegalArgumentException if the Locale has no date/time
  151.      *  pattern defined
  152.      */
  153.     // package protected, for access from FastDateFormat; do not make public or protected
  154.     F getDateTimeInstance(final int dateStyle, final int timeStyle, final TimeZone timeZone, final Locale locale) {
  155.         return getDateTimeInstance(Integer.valueOf(dateStyle), Integer.valueOf(timeStyle), timeZone, locale);
  156.     }

  157.     /**
  158.      * Gets a date/time formatter instance using the specified style,
  159.      * time zone and locale.
  160.      *
  161.      * @param dateStyle  date style: FULL, LONG, MEDIUM, or SHORT, null indicates no date in format
  162.      * @param timeStyle  time style: FULL, LONG, MEDIUM, or SHORT, null indicates no time in format
  163.      * @param timeZone  optional time zone, overrides time zone of
  164.      *  formatted date, null means use default Locale
  165.      * @param locale  optional locale, overrides system locale
  166.      * @return a localized standard date/time formatter
  167.      * @throws IllegalArgumentException if the Locale has no date/time
  168.      *  pattern defined
  169.      */
  170.     // This must remain private, see LANG-884
  171.     private F getDateTimeInstance(final Integer dateStyle, final Integer timeStyle, final TimeZone timeZone, Locale locale) {
  172.         locale = LocaleUtils.toLocale(locale);
  173.         final String pattern = getPatternForStyle(dateStyle, timeStyle, locale);
  174.         return getInstance(pattern, timeZone, locale);
  175.     }

  176.     /**
  177.      * Gets a formatter instance using the default pattern in the
  178.      * default time zone and locale.
  179.      *
  180.      * @return a date/time formatter
  181.      */
  182.     public F getInstance() {
  183.         return getDateTimeInstance(DateFormat.SHORT, DateFormat.SHORT, TimeZone.getDefault(), Locale.getDefault());
  184.     }

  185.     /**
  186.      * Gets a formatter instance using the specified pattern, time zone
  187.      * and locale.
  188.      *
  189.      * @param pattern  {@link java.text.SimpleDateFormat} compatible
  190.      *  pattern, non-null
  191.      * @param timeZone  the time zone, null means use the default TimeZone
  192.      * @param locale  the locale, null means use the default Locale
  193.      * @return a pattern based date/time formatter
  194.      * @throws NullPointerException if pattern is {@code null}
  195.      * @throws IllegalArgumentException if pattern is invalid
  196.      */
  197.     public F getInstance(final String pattern, final TimeZone timeZone, final Locale locale) {
  198.         Objects.requireNonNull(pattern, "pattern");
  199.         final TimeZone actualTimeZone = TimeZones.toTimeZone(timeZone);
  200.         final Locale actualLocale = LocaleUtils.toLocale(locale);
  201.         final ArrayKey key = new ArrayKey(pattern, actualTimeZone, actualLocale);
  202.         return cInstanceCache.computeIfAbsent(key, k -> createInstance(pattern, actualTimeZone, actualLocale));
  203.     }

  204.     /**
  205.      * Gets a time formatter instance using the specified style,
  206.      * time zone and locale.
  207.      *
  208.      * @param timeStyle  time style: FULL, LONG, MEDIUM, or SHORT
  209.      * @param timeZone  optional time zone, overrides time zone of
  210.      *  formatted date, null means use default Locale
  211.      * @param locale  optional locale, overrides system locale
  212.      * @return a localized standard date/time formatter
  213.      * @throws IllegalArgumentException if the Locale has no date/time
  214.      *  pattern defined
  215.      */
  216.     // package protected, for access from FastDateFormat; do not make public or protected
  217.     F getTimeInstance(final int timeStyle, final TimeZone timeZone, final Locale locale) {
  218.         return getDateTimeInstance(null, Integer.valueOf(timeStyle), timeZone, locale);
  219.     }

  220. }