If there is issue connected to dates in multi-threaded application and first to check is: do it use SimpleDateFormat.
Why you should never use SimpleDataFormat? It's not safe!
.. and here is proof:
Run this test class and you will see (as is not predictable run it few times):
/** * Please feel free to experiment - not only wrong data but sometimes number format exceptions... */ public class SimpleDateTest { static SimpleDateFormat df = new SimpleDateFormat("dd-MMM-yyyy"); static String testdata[] = { "01-Jan-1999", "14-Feb-2001", "31-Dec-2007" }; /** * Test method for SDF. */ @Test public void testParse() { Runnable r[] = new Runnable[testdata.length]; for (int i = 0; i < r.length; i++) { final int i2 = i; r[i] = new Runnable() { public void run() { try { for (int j = 0; j < 1000; j++) { String str = testdata[i2]; String str2 = null; // synchronized(df) { Date d = df.parse(str); str2 = df.format(d); } Assert.assertEquals("date conversion failed after " + j + " iterations.", str, str2); } } catch (ParseException e) { throw new RuntimeException("parse failed"); } } }; new Thread(r[i]).start(); } } }
Possible outputs are:
Exception in thread "Thread-0" junit.framework.ComparisonFailure: date conversion failed after 0 iterations. expected:<[01-Jan-1999]> but was:<[14-Feb-2001]> at junit.framework.Assert.assertEquals(Assert.java:85)
Exception in thread "Thread-0" Exception in thread "Thread-1" java.lang.NumberFormatException: For input string: "19992001.E199920014E"
Exception in thread "Thread-0" java.lang.NumberFormatException: multiple points at sun.misc.FloatingDecimal.readJavaFormatString(Unknown Source)
Exception in thread "Thread-0" java.lang.NumberFormatException: For input string: "" at java.lang.NumberFormatException.forInputString(Unknown Source)
And solution is synchronize usage of SimpleDateFormat, use ThreadLocal like here: ThreadSafeSimpleDateFormat:
tsd
or my last findings use improved version of ThreadLocal that creates HashMaps of SDF inside: SafeSimpleDateFormat
import java.text.DateFormatSymbols; import java.text.NumberFormat; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; import java.util.HashMap; import java.util.Map; import java.util.TimeZone; /** * 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 (jdereg@gmail.com) * <br/> * Copyright (c) John DeRegnaucourt * <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 { private final String _format; private static final ThreadLocal<Map<String, SimpleDateFormat>> _dateFormats = new ThreadLocal<Map<String, SimpleDateFormat>>() { public Map<String, SimpleDateFormat> initialValue() { return new HashMap<String, SimpleDateFormat>(); } }; private 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 String format(Date date) { return getDateFormat(_format).format(date); } public String format(Object date) { return getDateFormat(_format).format(date); } 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); } }
No comments:
Post a Comment