simpleDateFormat是咱們比較經常使用的日期轉換類,可是它是一個線程不安全的類。 舉例證實java
public class DateFormatExample1 { //請求總數 private static int clientTotal = 1000; //同時容許執行的線程總數 private static int threadTotal = 20; private static SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); public static void main(String[] args) throws InterruptedException { final Semaphore semaphore = new Semaphore(threadTotal); final CountDownLatch countDownLatch = new CountDownLatch(clientTotal); ExecutorService executorService = Executors.newCachedThreadPool(); for (int i = 0; i < clientTotal; i++) { executorService.execute(new Runnable() { @Override public void run() { try { semaphore.acquire(); try { simpleDateFormat.parse("2018-09-26"); } catch (ParseException e) { e.printStackTrace(); } semaphore.release(); } catch (InterruptedException e) { e.printStackTrace(); } countDownLatch.countDown(); } }); } countDownLatch.await(); executorService.shutdown(); } }
這段代碼在運行的過程當中,會報錯。git
Exception in thread "pool-1-thread-2" Exception in thread "pool-1-thread-1" Exception in thread "pool-1-thread-8" Exception in thread "pool-1-thread-9" java.lang.NumberFormatException: For input string: "E.199E1" at sun.misc.FloatingDecimal.readJavaFormatString(Unknown Source) at sun.misc.FloatingDecimal.parseDouble(Unknown Source) at java.lang.Double.parseDouble(Unknown Source) at java.text.DigitList.getDouble(Unknown Source) at java.text.DecimalFormat.parse(Unknown Source) at java.text.SimpleDateFormat.subParse(Unknown Source) at java.text.SimpleDateFormat.parse(Unknown Source) at java.text.DateFormat.parse(Unknown Source) at com.guoy.concurrency.commonUnsafe.DateFormatExample1$1.run(DateFormatExample1.java:31) at java.util.concurrent.ThreadPoolExecutor.runWorker(Unknown Source) at java.util.concurrent.ThreadPoolExecutor$Worker.run(Unknown Source) at java.lang.Thread.run(Unknown Source) java.lang.NumberFormatException: multiple points at sun.misc.FloatingDecimal.readJavaFormatString(Unknown Source) at sun.misc.FloatingDecimal.parseDouble(Unknown Source) at java.lang.Double.parseDouble(Unknown Source)
定義一個靜態的SimpleDateFormat實例,是你們在日期工具類中比較一般的寫法,可是這種寫法在多線程的狀況下就會拋出異常。解決方案有不少種:安全
舉例:多線程
package com.guoy.concurrency.commonUnsafe; 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; //thread not safe public class DateFormatExample3 { //請求總數 private static int clientTotal = 1000; //同時容許執行的線程總數 private static int threadTotal = 20; private static ThreadLocal<SimpleDateFormat> simpleDateFormat = new ThreadLocal<SimpleDateFormat>() { @Override protected SimpleDateFormat initialValue() { return new SimpleDateFormat("yyyy-MM-dd"); } }; public static void main(String[] args) throws InterruptedException { final Semaphore semaphore = new Semaphore(threadTotal); final CountDownLatch countDownLatch = new CountDownLatch(clientTotal); ExecutorService executorService = Executors.newCachedThreadPool(); for (int i = 0; i < clientTotal; i++) { executorService.execute(new Runnable() { @Override public void run() { try { semaphore.acquire(); try { simpleDateFormat.get().parse("2018-09-26"); } catch (ParseException e) { e.printStackTrace(); } semaphore.release(); } catch (InterruptedException e) { e.printStackTrace(); } countDownLatch.countDown(); } }); } countDownLatch.await(); executorService.shutdown(); } }
引入maven依賴併發
<dependency> <groupId>joda-time</groupId> <artifactId>joda-time</artifactId> <version>2.9.9</version> </dependency>
import org.joda.time.DateTime; import org.joda.time.format.DateTimeFormat; import org.joda.time.format.DateTimeFormatter; public class DateFormatExample2 { //請求總數 private static int clientTotal = 1000; //同時容許執行的線程總數 private static int threadTotal = 20; private static DateTimeFormatter dateTimeFormatter = DateTimeFormat.forPattern("yyyy-MM-dd"); public static void main(String[] args) throws InterruptedException { final Semaphore semaphore = new Semaphore(threadTotal); final CountDownLatch countDownLatch = new CountDownLatch(clientTotal); ExecutorService executorService = Executors.newCachedThreadPool(); for (int i = 0; i < clientTotal; i++) { executorService.execute(new Runnable() { @Override public void run() { try { semaphore.acquire(); DateTime.parse("2018-12-04", dateTimeFormatter).toDate(); semaphore.release(); } catch (InterruptedException e) { e.printStackTrace(); } countDownLatch.countDown(); } }); } countDownLatch.await(); executorService.shutdown(); }