我想测试一个修复,其中SimpleDateFormat用作静态类变量。 我得到了
以前我的代码是
public class Abc{
private static SimpleDateFormat dateformatter;
public static String method1(final Calendar calendar) {
String thePattern = "ddMMM";
dateformatter = new SimpleDateFormat(thePattern, Locale.US);
sPars = dateformatter.format(calendar.getTime());
//Something
}
public static String method2(final Calendar calendar) {
String thePattern = "ddMMyyyy";
dateformatter = new SimpleDateFormat(thePattern, Locale.US);
sPars = dateformatter.format(calendar.getTime());
//Something
}
}
对于这个,我正处于异常
之下java.lang.ArrayIndexOutOfBoundsException: 965
at sun.util.calendar.BaseCalendar.getCalendarDateFromFixedDate(BaseCalendar.java:454)
at java.util.GregorianCalendar.computeFields(GregorianCalendar.java:2333)
at java.util.GregorianCalendar.computeFields(GregorianCalendar.java:2248)
at java.util.Calendar.setTimeInMillis(Calendar.java:1140)
at java.util.Calendar.setTime(Calendar.java:1106)
at java.text.SimpleDateFormat.format(SimpleDateFormat.java:955)
at java.text.SimpleDateFormat.format(SimpleDateFormat.java:948)
at java.text.DateFormat.format(DateFormat.java:336)
现在我已将其更改为:
public class Abc{
public static String method1(final Calendar calendar) {
String thePattern = "ddMMM";
SimpleDateFormat dateformatter = new SimpleDateFormat(thePattern, Locale.US);
sPars = dateformatter.format(calendar.getTime());
//Something
}
public static String method2(final Calendar calendar) {
String thePattern = "ddMMyyyy";
SimpleDateFormat dateformatter = new SimpleDateFormat(thePattern, Locale.US);
sPars = dateformatter.format(calendar.getTime());
//Something
}
}
如何测试第二个是否正确? JUnits有什么方法可以测试多线程吗?
答案 0 :(得分:5)
SimpleDateFormat不是线程安全的,如javadoc中所述:
日期格式未同步。建议为每个线程创建单独的格式实例。如果多个线程同时访问格式,则必须在外部进行同步。
但是在这里,你甚至因为:
而弄乱参考文献dateformatter = new SimpleDateFormat(thePattern, Locale.US);
由于dateformatter是静态的,因此它的内容在线程之间共享。
在本地创建一个新实例(就像之后一样)是一个很好的方法。线程之间不共享本地实例,也不会混淆多线程。
请不要忘记删除private static SimpleDateFormat dateformatter;
以避免混淆。
如果您只想为每个线程创建一个实例,也可以使用ThreadLocal变量:
private static final ThreadLocal<SimpleDateFormat> dateformatter1 =
new ThreadLocal<SimpleDateFormat>() {
@Override protected SimpleDateFormat initialValue() {
return new SimpleDateFormat("ddMMM");
}
};
private static final ThreadLocal<SimpleDateFormat> dateformatter2 =
new ThreadLocal<SimpleDateFormat>() {
@Override protected SimpleDateFormat initialValue() {
return new SimpleDateFormat("ddMMyyyy");
}
};
public static String method1(final Calendar calendar) {
SimpleDateFormat dateformatter = dateformatter1.get();
sPars = dateformatter.format(calendar.getTime());
//Something
}
public static String method2(final Calendar calendar) {
SimpleDateFormat dateformatter = dateformatter2.get();
sPars = dateformatter.format(calendar.getTime());
//Something
}