SimpleDateFormat類的線程安全問題和解決方案(解決方案篇一)

      網友投稿 638 2025-03-31

      大家好,我是冰河~~

      解決SimpledateFormat類在高并發場景下的線程安全問題可以有多種方式,這里,就列舉幾個常用的方式供參考,大家也可以在評論區給出更多的解決方案。

      SimpledateFormat類的線程安全問題和解決方案(問題篇)

      1.局部變量法

      最簡單的一種方式就是將SimpleDateFormat類對象定義成局部變量,如下所示的代碼,將SimpleDateFormat類對象定義在parse(String)方法的上面,即可解決問題。

      package io.binghe.concurrent.lab06; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Semaphore; /** * @author binghe * @version 1.0.0 * @description 局部變量法解決SimpleDateFormat類的線程安全問題 */ public class SimpleDateFormatTest02 { //執行總次數 private static final int EXECUTE_COUNT = 1000; //同時運行的線程數量 private static final int THREAD_COUNT = 20; public static void main(String[] args) throws InterruptedException { final Semaphore semaphore = new Semaphore(THREAD_COUNT); final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT); ExecutorService executorService = Executors.newCachedThreadPool(); for (int i = 0; i < EXECUTE_COUNT; i++){ executorService.execute(() -> { try { semaphore.acquire(); try { SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); simpleDateFormat.parse("2020-01-01"); } catch (ParseException e) { System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗"); e.printStackTrace(); System.exit(1); }catch (NumberFormatException e){ System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗"); e.printStackTrace(); System.exit(1); } semaphore.release(); } catch (InterruptedException e) { System.out.println("信號量發生錯誤"); e.printStackTrace(); System.exit(1); } countDownLatch.countDown(); }); } countDownLatch.await(); executorService.shutdown(); System.out.println("所有線程格式化日期成功"); } }

      此時運行修改后的程序,輸出結果如下所示。

      所有線程格式化日期成功

      SimpleDateFormat類的線程安全問題和解決方案(解決方案篇一)

      至于在高并發場景下使用局部變量為何能解決線程的安全問題,會在【JVM專題】的JVM內存模式相關內容中深入剖析,這里不做過多的介紹了。

      當然,這種方式在高并發下會創建大量的SimpleDateFormat類對象,影響程序的性能,所以,這種方式在實際生產環境不太被推薦。

      2.synchronized鎖方式

      將SimpleDateFormat類對象定義成全局靜態變量,此時所有線程共享SimpleDateFormat類對象,此時在調用格式化時間的方法時,對SimpleDateFormat對象進行同步即可,代碼如下所示。

      package io.binghe.concurrent.lab06; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Semaphore; /** * @author binghe * @version 1.0.0 * @description 通過Synchronized鎖解決SimpleDateFormat類的線程安全問題 */ public class SimpleDateFormatTest03 { //執行總次數 private static final int EXECUTE_COUNT = 1000; //同時運行的線程數量 private static final int THREAD_COUNT = 20; //SimpleDateFormat對象 private static SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); public static void main(String[] args) throws InterruptedException { final Semaphore semaphore = new Semaphore(THREAD_COUNT); final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT); ExecutorService executorService = Executors.newCachedThreadPool(); for (int i = 0; i < EXECUTE_COUNT; i++){ executorService.execute(() -> { try { semaphore.acquire(); try { synchronized (simpleDateFormat){ simpleDateFormat.parse("2020-01-01"); } } catch (ParseException e) { System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗"); e.printStackTrace(); System.exit(1); }catch (NumberFormatException e){ System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗"); e.printStackTrace(); System.exit(1); } semaphore.release(); } catch (InterruptedException e) { System.out.println("信號量發生錯誤"); e.printStackTrace(); System.exit(1); } countDownLatch.countDown(); }); } countDownLatch.await(); executorService.shutdown(); System.out.println("所有線程格式化日期成功"); } }

      此時,解決問題的關鍵代碼如下所示。

      synchronized (simpleDateFormat){ simpleDateFormat.parse("2020-01-01"); }

      運行程序,輸出結果如下所示。

      所有線程格式化日期成功

      需要注意的是,雖然這種方式能夠解決SimpleDateFormat類的線程安全問題,但是由于在程序的執行過程中,為SimpleDateFormat類對象加上了synchronized鎖,導致同一時刻只能有一個線程執行parse(String)方法。此時,會影響程序的執行性能,在要求高并發的生產環境下,此種方式也是不太推薦使用的。

      3.Lock鎖方式

      Lock鎖方式與synchronized鎖方式實現原理相同,都是在高并發下通過JVM的鎖機制來保證程序的線程安全。通過Lock鎖方式解決問題的代碼如下所示。

      package io.binghe.concurrent.lab06; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Semaphore; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; /** * @author binghe * @version 1.0.0 * @description 通過Lock鎖解決SimpleDateFormat類的線程安全問題 */ public class SimpleDateFormatTest04 { //執行總次數 private static final int EXECUTE_COUNT = 1000; //同時運行的線程數量 private static final int THREAD_COUNT = 20; //SimpleDateFormat對象 private static SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); //Lock對象 private static Lock lock = new ReentrantLock(); public static void main(String[] args) throws InterruptedException { final Semaphore semaphore = new Semaphore(THREAD_COUNT); final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT); ExecutorService executorService = Executors.newCachedThreadPool(); for (int i = 0; i < EXECUTE_COUNT; i++){ executorService.execute(() -> { try { semaphore.acquire(); try { lock.lock(); simpleDateFormat.parse("2020-01-01"); } catch (ParseException e) { System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗"); e.printStackTrace(); System.exit(1); }catch (NumberFormatException e){ System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗"); e.printStackTrace(); System.exit(1); }finally { lock.unlock(); } semaphore.release(); } catch (InterruptedException e) { System.out.println("信號量發生錯誤"); e.printStackTrace(); System.exit(1); } countDownLatch.countDown(); }); } countDownLatch.await(); executorService.shutdown(); System.out.println("所有線程格式化日期成功"); } }

      通過代碼可以得知,首先,定義了一個Lock類型的全局靜態變量作為加鎖和釋放鎖的句柄。然后在simpleDateFormat.parse(String)代碼之前通過lock.lock()加鎖。這里需要注意的一點是:為防止程序拋出異常而導致鎖不能被釋放,一定要將釋放鎖的操作放到finally代碼塊中,如下所示。

      finally { lock.unlock(); }

      運行程序,輸出結果如下所示。

      所有線程格式化日期成功

      此種方式同樣會影響高并發場景下的性能,不太建議在高并發的生產環境使用。

      好了,今天就到這兒吧,剩下的解決方案我們下一篇再探討,我是冰河,我們下期見~~

      Java JDK 任務調度 多線程

      版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。

      版權聲明:本文內容由網絡用戶投稿,版權歸原作者所有,本站不擁有其著作權,亦不承擔相應法律責任。如果您發現本站中有涉嫌抄襲或描述失實的內容,請聯系我們jiasou666@gmail.com 處理,核實后本網站將在24小時內刪除侵權內容。

      上一篇:低代碼時代的呼叫中心(六)
      下一篇:破解行業痛點,華為云助力開天創世打開游戲市場新局面
      相關文章
      亚洲AV综合色区无码二区偷拍| 亚洲视频国产视频| 亚洲制服在线观看| 亚洲AV无码精品色午夜果冻不卡 | 亚洲国产精品成人AV在线| 亚洲第一香蕉视频| 亚洲黄色免费网站| 亚洲国产女人aaa毛片在线| 亚洲AV无码成人精品区天堂| 亚洲日韩精品一区二区三区| 国产亚洲精品看片在线观看| 国产精品亚洲综合专区片高清久久久 | 亚洲中文字幕在线第六区| 亚洲男女内射在线播放| 亚洲国产成人久久综合碰| 亚洲另类少妇17p| 色噜噜AV亚洲色一区二区| 亚洲人成在线播放网站| 成人午夜亚洲精品无码网站| 亚洲Av无码国产情品久久| 亚洲国产精品丝袜在线观看| 亚洲日本中文字幕天堂网| 国产亚洲精品成人a v小说| 亚洲日韩小电影在线观看| 亚洲成在人线av| 亚洲成a人片在线观看中文动漫 | 亚洲色偷偷综合亚洲av78 | 亚洲夜夜欢A∨一区二区三区| 亚洲最大AV网站在线观看| 亚洲自偷自偷偷色无码中文| 久久亚洲精品成人综合| 亚洲视频网站在线观看| 亚洲国产成人久久77| 中文有码亚洲制服av片| 亚洲国产午夜精品理论片在线播放| 亚洲AV无码一区二区三区网址| vvvv99日韩精品亚洲| 国产偷窥女洗浴在线观看亚洲| 亚洲日韩精品无码一区二区三区| 亚洲AV日韩AV天堂久久| 亚洲国产夜色在线观看|