我有许多Java日期模式,并希望通过静态引用在不同线程的SimpleDateFormat
对象中多次重用它们,以提高速度。
代码看起来像这样(在一个名为 FormatClass
的类中):
private static String[] PATTERNS = new String[] {...};
public ThreadLocal<SimpleDateFormat[]> LOCAL_FORMATS = new ThreadLocal<SimpleDateFormat[]>
{
@Override
protected SimpleDateFormat[] initialValue()
{
List<SimpleDateFormat> formatList = new ArrayList<SimpleDateFormat>();
for (String pattern:PATTERNS)
{
formatList.add(new SimpleDateFormat(pattern);
}
return formatList.toArray(new SimpleDateFormat[0]);
}
}
使用上面的代码,另一个类上的方法可以format
(或parse
)多个日期字符串,如下所示:
public static void printFormatted(String date)
{
for (SimpleDateFormat sdf:FormatClass.LOCAL_FORMATS.get())
{
System.out.println(sdf.format(date));
}
}
其中printFormatted()
方法可能是静态的,也可能不是静态的,但肯定会被多个不同的线程访问。
上述方法会按预期工作吗?
直接回答您的问题:是的,每个线程都会有自己独特的副本,如 SimpleDateFormat 文档中的建议。
它看起来不错,但您可以使用多线程程序轻松测试它。只需打印线程的 Id ( Thread.currentThread().toString()
) 并System.identifyHashCode(simpleDateFormat)
并验证您是否为每个线程获取了唯一的副本并适当地重用它们。
这很好用,但代码相对复杂。我建议检查一下,也许您可以将 SimpleDateFormat 分配给方法中的某个局部变量(始终是线程本地)。当许多值在一个循环中格式化时,这可以很好地工作,然后可以在同一方法中。
一种更优雅的方法,还没有人提出过——只需为 SimpleDateFormat 创建一个插入式替换包装类,以同步对其方法的访问。 在大多数情况下,我认为同步开销并不比 ThreadLocalMap 中的哈希查找差。