一、線程安全問題產(chǎn)生的原因
線程安全問題都是由全局變量及靜態(tài)變量引起的
二、線程安全問題
SimpleDateFormate sdf = new SimpleDateFormat();使用sdf.parse(dateStr);sdf.format(date);在sdf內(nèi)有一個(gè)對Caleadar對象的引用,在源碼sdf.parse(dateStr);源碼中calendar.clear();和calendar.getTime(); // 獲取calendar的時(shí)間
如果 線程A 調(diào)用了 sdf.parse(), 并且進(jìn)行了 calendar.clear()后還未執(zhí)行calendar.getTime()的時(shí)候,線程B又調(diào)用了sdf.parse(), 這時(shí)候線程B也執(zhí)行了sdf.clear()方法, 這樣就導(dǎo)致線程A的的calendar數(shù)據(jù)被清空了;
ThreadLocal是使用空間換時(shí)間,synchronized是使用時(shí)間換空間
使用ThreadLocal解決線程安全:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
public class ThreadLocalDateUtil { private static final String date_format = "yyyy-MM-dd HH:mm:ss" ; private static ThreadLocal<DateFormat> threadLocal = new ThreadLocal<DateFormat>(); public static DateFormat getDateFormat() { DateFormat df = threadLocal.get(); if (df== null ){ df = new SimpleDateFormat(date_format); threadLocal.set(df); } return df; } public static String formatDate(Date date) throws ParseException { return getDateFormat().format(date); } public static Date parse(String strDate) throws ParseException { return getDateFormat().parse(strDate); } } |
使用synchronized解決方案:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public class DateSyncUtil { private static SimpleDateFormat sdf = new SimpleDateFormat( "yyyy-MM-dd HH:mm:ss" ); public static String formatDate(Date date) throws ParseException{ synchronized (sdf){ return sdf.format(date); } } public static Date parse(String strDate) throws ParseException{ synchronized (sdf){ return sdf.parse(strDate); } } } |
感謝閱讀本文,希望能幫助到大家,謝謝大家對本站的支持!