LANG-829: FastDateParser could use Calendar.getDisplayNames for all text fields

git-svn-id: https://svn.apache.org/repos/asf/commons/proper/lang/trunk@1390839 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Chas Honton 2012-09-27 04:50:21 +00:00
parent 9960dd4204
commit 8c87749695
1 changed files with 70 additions and 132 deletions

View File

@ -19,7 +19,6 @@ package org.apache.commons.lang3.time;
import java.io.IOException; import java.io.IOException;
import java.io.ObjectInputStream; import java.io.ObjectInputStream;
import java.io.Serializable; import java.io.Serializable;
import java.text.DateFormatSymbols;
import java.text.ParseException; import java.text.ParseException;
import java.text.ParsePosition; import java.text.ParsePosition;
import java.util.ArrayList; import java.util.ArrayList;
@ -73,9 +72,6 @@ public class FastDateParser implements DateParser, Serializable {
*/ */
private static final long serialVersionUID = 1L; private static final long serialVersionUID = 1L;
private static final ConcurrentMap<Locale,TimeZoneStrategy> tzsCache=
new ConcurrentHashMap<Locale,TimeZoneStrategy>(3);
static final Locale JAPANESE_IMPERIAL = new Locale("ja","JP","JP"); static final Locale JAPANESE_IMPERIAL = new Locale("ja","JP","JP");
// defining fields // defining fields
@ -87,7 +83,6 @@ public class FastDateParser implements DateParser, Serializable {
private transient Pattern parsePattern; private transient Pattern parsePattern;
private transient Strategy[] strategies; private transient Strategy[] strategies;
private transient int thisYear; private transient int thisYear;
private transient ConcurrentMap<Integer, KeyValue[]> nameValues;
// dynamic fields to communicate with Strategy // dynamic fields to communicate with Strategy
private transient String currentFormatField; private transient String currentFormatField;
@ -113,9 +108,8 @@ public class FastDateParser implements DateParser, Serializable {
* This is called from constructor and from readObject (de-serialization) * This is called from constructor and from readObject (de-serialization)
*/ */
private void init() { private void init() {
thisYear= Calendar.getInstance(timeZone, locale).get(Calendar.YEAR); Calendar definingCalendar = Calendar.getInstance(timeZone, locale);
thisYear= definingCalendar.get(Calendar.YEAR);
nameValues= new ConcurrentHashMap<Integer, KeyValue[]>();
StringBuilder regex= new StringBuilder(); StringBuilder regex= new StringBuilder();
List<Strategy> collector = new ArrayList<Strategy>(); List<Strategy> collector = new ArrayList<Strategy>();
@ -127,7 +121,7 @@ public class FastDateParser implements DateParser, Serializable {
} }
currentFormatField= patternMatcher.group(); currentFormatField= patternMatcher.group();
Strategy currentStrategy= getStrategy(currentFormatField); Strategy currentStrategy= getStrategy(currentFormatField, definingCalendar);
for(;;) { for(;;) {
patternMatcher.region(patternMatcher.end(), patternMatcher.regionEnd()); patternMatcher.region(patternMatcher.end(), patternMatcher.regionEnd());
if(!patternMatcher.lookingAt()) { if(!patternMatcher.lookingAt()) {
@ -135,7 +129,7 @@ public class FastDateParser implements DateParser, Serializable {
break; break;
} }
String nextFormatField= patternMatcher.group(); String nextFormatField= patternMatcher.group();
nextStrategy = getStrategy(nextFormatField); nextStrategy = getStrategy(nextFormatField, definingCalendar);
if(currentStrategy.addRegex(this, regex)) { if(currentStrategy.addRegex(this, regex)) {
collector.add(currentStrategy); collector.add(currentStrategy);
} }
@ -368,111 +362,28 @@ public class FastDateParser implements DateParser, Serializable {
/** /**
* Get the short and long values displayed for a field * Get the short and long values displayed for a field
* @param field The field of interest * @param field The field of interest
* @param definingCalendar The calendar to obtain the short and long values
* @param locale The locale of dislay names
* @return A sorted array of the field key / value pairs * @return A sorted array of the field key / value pairs
*/ */
KeyValue[] getDisplayNames(int field) { private static KeyValue[] getDisplayNames(int field, Calendar definingCalendar, Locale locale) {
Integer fieldInt = Integer.valueOf(field); List<KeyValue> keyValues = new ArrayList<KeyValue>(24);
KeyValue[] fieldKeyValues= nameValues.get(fieldInt); addNamesToKeyValues(keyValues, definingCalendar.getDisplayNames(field, Calendar.ALL_STYLES, locale));
if(fieldKeyValues==null) { return createKeyValues(keyValues);
DateFormatSymbols symbols= DateFormatSymbols.getInstance(locale);
switch(field) {
case Calendar.ERA:
// DateFormatSymbols#getEras() only returns AD/BC or translations
// It does not work for the Thai Buddhist or Japanese Imperial calendars.
// see: https://issues.apache.org/jira/browse/TRINIDAD-2126
Calendar c = Calendar.getInstance(locale);
// N.B. Some calendars have different short and long symbols, e.g. ja_JP_JP
// TODO Seems to be only that locale; if that is guaranteed, could skip some work here
String[] shortEras = toArray(c.getDisplayNames(Calendar.ERA, Calendar.SHORT, locale));
String[] longEras = toArray(c.getDisplayNames(Calendar.ERA, Calendar.LONG, locale));
if (Arrays.equals(shortEras, longEras)) {
fieldKeyValues = createKeyValues(longEras, null); // save memory
} else {
fieldKeyValues = createKeyValues(longEras, shortEras);
}
break;
case Calendar.DAY_OF_WEEK:
fieldKeyValues= createKeyValues(symbols.getWeekdays(), symbols.getShortWeekdays());
break;
case Calendar.AM_PM:
fieldKeyValues= createKeyValues(symbols.getAmPmStrings(), null);
break;
case Calendar.MONTH:
fieldKeyValues= createKeyValues(symbols.getMonths(), symbols.getShortMonths());
break;
default:
throw new IllegalArgumentException("Invalid field value "+field);
}
KeyValue[] prior = nameValues.putIfAbsent(fieldInt, fieldKeyValues);
if(prior!=null) {
fieldKeyValues= prior;
}
}
return fieldKeyValues;
} }
private String[] toArray(Map<String, Integer> era) { private static void addNamesToKeyValues(List<KeyValue> keyValues, Map<String, Integer> displayNames) {
String[] eras = new String[era.size()]; // assume no gaps in entry values for(Map.Entry<String, Integer> me : displayNames.entrySet()) {
for(Map.Entry<String, Integer> me : era.entrySet()) { keyValues.add(new KeyValue(me.getKey(), me.getValue()));
int idx = me.getValue().intValue();
final String key = me.getKey();
if (key == null) {
throw new IllegalArgumentException();
}
eras[idx] = key;
} }
return eras;
} }
/** private static KeyValue[] createKeyValues(List<KeyValue> keyValues) {
* Create key / value pairs from keys KeyValue[] fieldKeyValues= keyValues.toArray(new KeyValue[keyValues.size()]);
* @param longValues The allowable long names for a field
* @param shortValues The optional allowable short names for a field
* @return The sorted name / value pairs for the field
*/
private static KeyValue[] createKeyValues(String[] longValues, String[] shortValues) {
KeyValue[] fieldKeyValues= new KeyValue[count(longValues)+count(shortValues)];
copy(fieldKeyValues, copy(fieldKeyValues, 0, longValues), shortValues);
Arrays.sort(fieldKeyValues, IGNORE_CASE_COMPARATOR); Arrays.sort(fieldKeyValues, IGNORE_CASE_COMPARATOR);
return fieldKeyValues; return fieldKeyValues;
} }
/**
* Get a count of valid values in array. A valid value is of non-zero length.
* @param values The values to check. This parameter may be null
* @return The number of valid values
*/
private static int count(String[] values) {
int count= 0;
if(values!=null) {
for(String value : values) {
if(value.length()>0) {
++count;
}
}
}
return count;
}
/**
* Create key / value pairs from values
* @param fieldKeyValues The destination array
* @param offset The offset into the destination array
* @param values The values to use to create key / value pairs. This parameter may be null.
* @return The offset into the destination array
*/
private static int copy(KeyValue[] fieldKeyValues, int offset, String[] values) {
if(values!=null) {
for(int i= 0; i<values.length; ++i) {
String value= values[i];
if(value.length()>0) {
fieldKeyValues[offset++]= new KeyValue(value, i);
}
}
}
return offset;
}
/** /**
* Adjust dates to be within 80 years before and 20 years after instantiation * Adjust dates to be within 80 years before and 20 years after instantiation
* @param twoDigitYear The year to adjust * @param twoDigitYear The year to adjust
@ -523,6 +434,7 @@ public class FastDateParser implements DateParser, Serializable {
* which will accept this field * which will accept this field
* @param parser The parser calling this strategy * @param parser The parser calling this strategy
* @param regex The <code>StringBuilder</code> to append to * @param regex The <code>StringBuilder</code> to append to
* @param Calendar The calendar this strategy must parse
* @return true, if this field will set the calendar; * @return true, if this field will set the calendar;
* false, if this field is a constant value * false, if this field is a constant value
*/ */
@ -538,9 +450,10 @@ public class FastDateParser implements DateParser, Serializable {
/** /**
* Obtain a Strategy given a field from a SimpleDateFormat pattern * Obtain a Strategy given a field from a SimpleDateFormat pattern
* @param formatField A sub-sequence of the SimpleDateFormat pattern * @param formatField A sub-sequence of the SimpleDateFormat pattern
* @param definingCalendar The calendar to obtain the short and long values
* @return The Strategy that will handle parsing for the field * @return The Strategy that will handle parsing for the field
*/ */
private Strategy getStrategy(String formatField) { private Strategy getStrategy(String formatField, Calendar definingCalendar) {
switch(formatField.charAt(0)) { switch(formatField.charAt(0)) {
case '\'': case '\'':
if(formatField.length()>2) { if(formatField.length()>2) {
@ -552,25 +465,23 @@ public class FastDateParser implements DateParser, Serializable {
case 'D': case 'D':
return DAY_OF_YEAR_STRATEGY; return DAY_OF_YEAR_STRATEGY;
case 'E': case 'E':
return DAY_OF_WEEK_STRATEGY; return getLocaleSpecificStrategy(Calendar.DAY_OF_WEEK, definingCalendar);
case 'F': case 'F':
return DAY_OF_WEEK_IN_MONTH_STRATEGY; return DAY_OF_WEEK_IN_MONTH_STRATEGY;
case 'G': case 'G':
return ERA_STRATEGY; return getLocaleSpecificStrategy(Calendar.ERA, definingCalendar);
case 'H': case 'H':
return MODULO_HOUR_OF_DAY_STRATEGY; return MODULO_HOUR_OF_DAY_STRATEGY;
case 'K': case 'K':
return HOUR_STRATEGY; return HOUR_STRATEGY;
case 'M': case 'M':
return formatField.length()>=3 ?TEXT_MONTH_STRATEGY :NUMBER_MONTH_STRATEGY; return formatField.length()>=3 ?getLocaleSpecificStrategy(Calendar.MONTH, definingCalendar) :NUMBER_MONTH_STRATEGY;
case 'S': case 'S':
return MILLISECOND_STRATEGY; return MILLISECOND_STRATEGY;
case 'W': case 'W':
return WEEK_OF_MONTH_STRATEGY; return WEEK_OF_MONTH_STRATEGY;
case 'Z':
break;
case 'a': case 'a':
return AM_PM_STRATEGY; return getLocaleSpecificStrategy(Calendar.AM_PM, definingCalendar);
case 'd': case 'd':
return DAY_OF_MONTH_STRATEGY; return DAY_OF_MONTH_STRATEGY;
case 'h': case 'h':
@ -585,21 +496,52 @@ public class FastDateParser implements DateParser, Serializable {
return WEEK_OF_YEAR_STRATEGY; return WEEK_OF_YEAR_STRATEGY;
case 'y': case 'y':
return formatField.length()>2 ?LITERAL_YEAR_STRATEGY :ABBREVIATED_YEAR_STRATEGY; return formatField.length()>2 ?LITERAL_YEAR_STRATEGY :ABBREVIATED_YEAR_STRATEGY;
case 'Z':
case 'z': case 'z':
break; return getLocaleSpecificStrategy(Calendar.ZONE_OFFSET, definingCalendar);
} }
TimeZoneStrategy tzs= tzsCache.get(locale); }
if(tzs==null) {
tzs= new TimeZoneStrategy(locale); @SuppressWarnings("unchecked")
TimeZoneStrategy inCache= tzsCache.putIfAbsent(locale, tzs); private static ConcurrentMap<Locale, Strategy>[] caches = new ConcurrentMap[Calendar.FIELD_COUNT];
/**
* Get a cache of Strategies for a particular field
* @param field The Calendar field
* @return a cache of Locale to Strategy
*/
private static ConcurrentMap<Locale, Strategy> getCache(int field) {
synchronized(caches) {
if(caches[field]==null) {
caches[field]= new ConcurrentHashMap<Locale,Strategy>(3);
}
return caches[field];
}
}
/**
* Construct a Strategy that parses a Text field
* @param locale The Locale of the TimeZone to parse
* @param field The Calendar field
* @param definingCalendar The calendar to obtain the short and long values
* @return a TextStrategy for the field and Locale
*/
private Strategy getLocaleSpecificStrategy(int field, Calendar definingCalendar) {
ConcurrentMap<Locale,Strategy> cache = getCache(field);
Strategy strategy= cache.get(field);
if(strategy==null) {
strategy= field==Calendar.ZONE_OFFSET
? new TimeZoneStrategy(locale)
: new TextStrategy(field, definingCalendar, locale);
Strategy inCache= cache.putIfAbsent(locale, strategy);
if(inCache!=null) { if(inCache!=null) {
return inCache; return inCache;
} }
} }
return tzs; return strategy;
} }
/** /**
* A strategy that copies the static or quoted field in the parsing pattern * A strategy that copies the static or quoted field in the parsing pattern
*/ */
private static class CopyQuotedStrategy implements Strategy { private static class CopyQuotedStrategy implements Strategy {
@ -642,18 +584,20 @@ public class FastDateParser implements DateParser, Serializable {
} }
} }
/** /**
* A strategy that handles a text field in the parsing pattern * A strategy that handles a text field in the parsing pattern
*/ */
private static class TextStrategy implements Strategy { private static class TextStrategy implements Strategy {
private final int field; private final int field;
private final KeyValue[] keyValues;
/** /**
* Construct a Strategy that parses a Text field * Construct a Strategy that parses a Text field
* @param field The Calendar field * @param field The Calendar field
*/ */
TextStrategy(int field) { TextStrategy(int field, Calendar definingCalendar, Locale locale) {
this.field= field; this.field= field;
this.keyValues= getDisplayNames(field, definingCalendar, locale);
} }
/** /**
@ -670,7 +614,7 @@ public class FastDateParser implements DateParser, Serializable {
@Override @Override
public boolean addRegex(FastDateParser parser, StringBuilder regex) { public boolean addRegex(FastDateParser parser, StringBuilder regex) {
regex.append('('); regex.append('(');
for(KeyValue textKeyValue : parser.getDisplayNames(field)) { for(KeyValue textKeyValue : keyValues) {
escapeRegex(regex, textKeyValue.key, false).append('|'); escapeRegex(regex, textKeyValue.key, false).append('|');
} }
regex.setCharAt(regex.length()-1, ')'); regex.setCharAt(regex.length()-1, ')');
@ -682,21 +626,21 @@ public class FastDateParser implements DateParser, Serializable {
*/ */
@Override @Override
public void setCalendar(FastDateParser parser, Calendar cal, String value) { public void setCalendar(FastDateParser parser, Calendar cal, String value) {
KeyValue[] textKeyValues= parser.getDisplayNames(field); int idx= Arrays.binarySearch(keyValues, new KeyValue(value, -1), IGNORE_CASE_COMPARATOR);
int idx= Arrays.binarySearch(textKeyValues, new KeyValue(value, -1), IGNORE_CASE_COMPARATOR);
if(idx<0) { if(idx<0) {
StringBuilder sb= new StringBuilder(value); StringBuilder sb= new StringBuilder(value);
sb.append(" not in ("); sb.append(" not in (");
for(KeyValue textKeyValue : textKeyValues) { for(KeyValue textKeyValue : keyValues) {
sb.append(textKeyValue.key).append(' '); sb.append(textKeyValue.key).append(' ');
} }
sb.setCharAt(sb.length()-1, ')'); sb.setCharAt(sb.length()-1, ')');
throw new IllegalArgumentException(sb.toString()); throw new IllegalArgumentException(sb.toString());
} }
cal.set(field, textKeyValues[idx].value); cal.set(field, keyValues[idx].value);
} }
} }
/** /**
* A strategy that handles a number field in the parsing pattern * A strategy that handles a number field in the parsing pattern
*/ */
@ -838,12 +782,6 @@ public class FastDateParser implements DateParser, Serializable {
} }
} }
private static final Strategy ERA_STRATEGY = new TextStrategy(Calendar.ERA);
private static final Strategy DAY_OF_WEEK_STRATEGY = new TextStrategy(Calendar.DAY_OF_WEEK);
private static final Strategy AM_PM_STRATEGY = new TextStrategy(Calendar.AM_PM);
private static final Strategy TEXT_MONTH_STRATEGY = new TextStrategy(Calendar.MONTH);
private static final Strategy NUMBER_MONTH_STRATEGY = new NumberStrategy(Calendar.MONTH) { private static final Strategy NUMBER_MONTH_STRATEGY = new NumberStrategy(Calendar.MONTH) {
@Override @Override
public int modify(int iValue) { public int modify(int iValue) {