package com.cedarsoftware.util; import java.text.DateFormatSymbols; import java.text.FieldPosition; import java.text.Format; import java.text.NumberFormat; import java.text.ParseException; import java.text.ParsePosition; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; import java.util.Map; import java.util.TimeZone; import java.util.concurrent.ConcurrentHashMap; /** * This class implements a Thread-Safe (re-entrant) SimpleDateFormat * class. It does this by using a ThreadLocal that holds a Map, instead * of the traditional approach to hold the SimpleDateFormat in a ThreadLocal. * * Each ThreadLocal holds a single HashMap containing SimpleDateFormats, keyed * by a String format (e.g. "yyyy/M/d", etc.), for each new SimpleDateFormat * instance that was created within the threads execution context. * * @author John DeRegnaucourt (john@cedarsoftware.com) * <br> * Copyright (c) Cedar Software LLC * <br><br> * Licensed 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 * <br><br> * http://www.apache.org/licenses/LICENSE-2.0 * <br><br> * 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. */ public class SafeSimpleDateFormat extends Format { private final String _format; private static final ThreadLocal<Map<String, SimpleDateFormat>> _dateFormats = new ThreadLocal<Map<String, SimpleDateFormat>>() { public Map<String, SimpleDateFormat> initialValue() { return new ConcurrentHashMap<>(); } }; public static SimpleDateFormat getDateFormat(String format) { Map<String, SimpleDateFormat> formatters = _dateFormats.get(); SimpleDateFormat formatter = formatters.get(format); if (formatter == null) { formatter = new SimpleDateFormat(format); formatters.put(format, formatter); } return formatter; } public SafeSimpleDateFormat(String format) { _format = format; } public StringBuffer format(Object obj, StringBuffer toAppendTo, FieldPosition pos) { return getDateFormat(_format).format(obj, toAppendTo, pos); } public Object parseObject(String source, ParsePosition pos) { return getDateFormat(_format).parse(source, pos); } public Date parse(String day) throws ParseException { return getDateFormat(_format).parse(day); } public void setTimeZone(TimeZone tz) { getDateFormat(_format).setTimeZone(tz); } public void setCalendar(Calendar cal) { getDateFormat(_format).setCalendar(cal); } public void setNumberFormat(NumberFormat format) { getDateFormat(_format).setNumberFormat(format); } public void setLenient(boolean lenient) { getDateFormat(_format).setLenient(lenient); } public void setDateFormatSymbols(DateFormatSymbols symbols) { getDateFormat(_format).setDateFormatSymbols(symbols); } public void set2DigitYearStart(Date date) { getDateFormat(_format).set2DigitYearStart(date); } }