一区二区三区电影_国产伦精品一区二区三区视频免费_亚洲欧美国产精品va在线观看_国产精品一二三四

聯系我們 - 廣告服務 - 聯系電話:
您的當前位置: > 關注 > > 正文

深入理解Java:SimpleDateFormat安全的時間格式化

來源:CSDN 時間:2023-03-14 08:43:45

深入理解Java:SimpleDateFormat安全的時間格式化

想必大家對SimpleDateFormat并不陌生。SimpleDateFormat 是 Java 中一個非常常用的類,該類用來對日期字符串進行解析和格式化輸出,但如果使用不小心會導致非常微妙和難以調試的問題,因為 DateFormat 和 SimpleDateFormat 類不都是線程安全的,在多線程環境下調用 format() 和 parse() 方法應該使用同步代碼來避免問題。下面我們通過一個具體的場景來一步步的深入學習和理解SimpleDateFormat類。

一.引子我們都是優秀的程序員,我們都知道在程序中我們應當盡量少的創建SimpleDateFormat 實例,因為創建這么一個實例需要耗費很大的代價。在一個讀取數據庫數據導出到excel文件的例子當中,每次處理一個時間信息的時候,就需要創建一個SimpleDateFormat實例對象,然后再丟棄這個對象。大量的對象就這樣被創建出來,占用大量的內存和 jvm空間。代碼如下:


(資料圖)

package com.peidasoft.dateformat;import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Date;public class DateUtil {        public static  String formatDate(Date date)throws ParseException{         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");        return sdf.format(date);    }        public static Date parse(String strDate) throws ParseException{         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");        return sdf.parse(strDate);    }}

你也許會說,OK,那我就創建一個靜態的simpleDateFormat實例,然后放到一個DateUtil類(如下)中,在使用時直接使用這個實例進行操作,這樣問題就解決了。改進后的代碼如下:

package com.peidasoft.dateformat;import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Date;public class DateUtil {    private static final  SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");        public static  String formatDate(Date date)throws ParseException{        return sdf.format(date);    }        public static Date parse(String strDate) throws ParseException{        return sdf.parse(strDate);    }}

當然,這個方法的確很不錯,在大部分的時間里面都會工作得很好。但當你在生產環境中使用一段時間之后,你就會發現這么一個事實:它不是線程安全的。在正常的測試情況之下,都沒有問題,但一旦在生產環境中一定負載情況下時,這個問題就出來了。他會出現各種不同的情況,比如轉化的時間不正確,比如報錯,比如線程被掛死等等。我們看下面的測試用例,那事實說話:

package com.peidasoft.dateformat;import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Date;public class DateUtil {        private static final  SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");        public static  String formatDate(Date date)throws ParseException{        return sdf.format(date);    }        public static Date parse(String strDate) throws ParseException{        return sdf.parse(strDate);    }}

package com.peidasoft.dateformat;import java.text.ParseException;import java.util.Date;public class DateUtilTest {        public static class TestSimpleDateFormatThreadSafe extends Thread {        @Override        public void run() {            while(true) {                try {                    this.join(2000);                } catch (InterruptedException e1) {                    e1.printStackTrace();                }                try {                    System.out.println(this.getName()+":"+DateUtil.parse("2013-05-24 06:02:20"));                } catch (ParseException e) {                    e.printStackTrace();                }            }        }        }            public static void main(String[] args) {        for(int i = 0; i < 3; i++){            new TestSimpleDateFormatThreadSafe().start();        }                }}

執行輸出如下:

Exception in thread "Thread-1" java.lang.NumberFormatException: multiple points    at sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1082)    at java.lang.Double.parseDouble(Double.java:510)    at java.text.DigitList.getDouble(DigitList.java:151)    at java.text.DecimalFormat.parse(DecimalFormat.java:1302)    at java.text.SimpleDateFormat.subParse(SimpleDateFormat.java:1589)    at java.text.SimpleDateFormat.parse(SimpleDateFormat.java:1311)    at java.text.DateFormat.parse(DateFormat.java:335)    at com.peidasoft.orm.dateformat.DateNoStaticUtil.parse(DateNoStaticUtil.java:17)    at com.peidasoft.orm.dateformat.DateUtilTest$TestSimpleDateFormatThreadSafe.run(DateUtilTest.java:20)Exception in thread "Thread-0" java.lang.NumberFormatException: multiple points    at sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1082)    at java.lang.Double.parseDouble(Double.java:510)    at java.text.DigitList.getDouble(DigitList.java:151)    at java.text.DecimalFormat.parse(DecimalFormat.java:1302)    at java.text.SimpleDateFormat.subParse(SimpleDateFormat.java:1589)    at java.text.SimpleDateFormat.parse(SimpleDateFormat.java:1311)    at java.text.DateFormat.parse(DateFormat.java:335)    at com.peidasoft.orm.dateformat.DateNoStaticUtil.parse(DateNoStaticUtil.java:17)    at com.peidasoft.orm.dateformat.DateUtilTest$TestSimpleDateFormatThreadSafe.run(DateUtilTest.java:20)Thread-2:Mon May 24 06:02:20 CST 2021Thread-2:Fri May 24 06:02:20 CST 2013Thread-2:Fri May 24 06:02:20 CST 2013Thread-2:Fri May 24 06:02:20 CST 2013

說明:Thread-1和Thread-0報java.lang.NumberFormatException: multiple points錯誤,直接掛死,沒起來;Thread-2 雖然沒有掛死,但輸出的時間是有錯誤的,比如我們輸入的時間是:2013-05-24 06:02:20 ,當會輸出:Mon May 24 06:02:20 CST 2021 這樣的靈異事件。

二.原因

作為一個專業程序員,我們當然都知道,相比于共享一個變量的開銷要比每次創建一個新變量要小很多。上面的優化過的靜態的SimpleDateFormat版,之所在并發情況下回出現各種靈異錯誤,是因為SimpleDateFormat和DateFormat類不是線程安全的。我們之所以忽視線程安全的問題,是因為從SimpleDateFormat和DateFormat類提供給我們的接口上來看,實在讓人看不出它與線程安全有何相干。只是在JDK文檔的最下面有如下說明:

SimpleDateFormat中的日期格式不是同步的。推薦(建議)為每個線程創建獨立的格式實例。如果多個線程同時訪問一個格式,則它必須保持外部同步。

JDK原始文檔如下:

Synchronization:

Date formats are not synchronized.

It is recommended to create separate format instances for each thread.

If multiple threads access a format concurrently, it must be synchronized externally.

下面我們通過看JDK源碼來看看為什么SimpleDateFormat和DateFormat類不是線程安全的真正原因:

SimpleDateFormat繼承了DateFormat,在DateFormat中定義了一個protected屬性的 Calendar類的對象:calendar。只是因為Calendar累的概念復雜,牽扯到時區與本地化等等,Jdk的實現中使用了成員變量來傳遞參數,這就造成在多線程的時候會出現錯誤。

在format方法里,有這樣一段代碼:

private StringBuffer format(Date date, StringBuffer toAppendTo,                                FieldDelegate delegate) {        // Convert input date to time field list        calendar.setTime(date);    boolean useDateFormatSymbols = useDateFormatSymbols();        for (int i = 0; i < compiledPattern.length; ) {            int tag = compiledPattern[i] >>> 8;        int count = compiledPattern[i++] & 0xff;        if (count == 255) {        count = compiledPattern[i++] << 16;        count |= compiledPattern[i++];        }        switch (tag) {        case TAG_QUOTE_ASCII_CHAR:        toAppendTo.append((char)count);        break;        case TAG_QUOTE_CHARS:        toAppendTo.append(compiledPattern, i, count);        i += count;        break;        default:                subFormat(tag, count, delegate, toAppendTo, useDateFormatSymbols);        break;        }    }        return toAppendTo;    }

calendar.setTime(date)這條語句改變了calendar,稍后,calendar還會用到(在subFormat方法里),而這就是引發問題的根源。想象一下,在一個多線程環境下,有兩個線程持有了同一個SimpleDateFormat的實例,分別調用format方法:

線程1調用format方法,改變了calendar這個字段。

中斷來了。

線程2開始執行,它也改變了calendar。

又中斷了。

線程1回來了,此時,calendar已然不是它所設的值,而是走上了線程2設計的道路。如果多個線程同時爭搶calendar對象,則會出現各種問題,時間不對,線程掛死等等。

分析一下format的實現,我們不難發現,用到成員變量calendar,唯一的好處,就是在調用subFormat時,少了一個參數,卻帶來了這許多的問題。其實,只要在這里用一個局部變量,一路傳遞下去,所有問題都將迎刃而解。

這個問題背后隱藏著一個更為重要的問題--無狀態:無狀態方法的好處之一,就是它在各種環境下,都可以安全的調用。衡量一個方法是否是有狀態的,就看它是否改動了其它的東西,比如全局變量,比如實例的字段。format方法在運行過程中改動了SimpleDateFormat的calendar字段,所以,它是有狀態的。

這也同時提醒我們在開發和設計系統的時候注意下一下三點:

1.自己寫公用類的時候,要對多線程調用情況下的后果在注釋里進行明確說明

2.對線程環境下,對每一個共享的可變變量都要注意其線程安全性

3.我們的類和方法在做設計的時候,要盡量設計成無狀態的

三.解決辦法

1.需要的時候創建新實例:

package com.peidasoft.dateformat;import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Date;public class DateUtil {        public static  String formatDate(Date date)throws ParseException{         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");        return sdf.format(date);    }        public static Date parse(String strDate) throws ParseException{         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");        return sdf.parse(strDate);    }}

說明:在需要用到SimpleDateFormat 的地方新建一個實例,不管什么時候,將有線程安全問題的對象由共享變為局部私有都能避免多線程問題,不過也加重了創建對象的負擔。在一般情況下,這樣其實對性能影響比不是很明顯的。

2.使用同步:同步SimpleDateFormat對象

package com.peidasoft.dateformat;import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Date;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);        }    } }

說明:當線程較多時,當一個線程調用該方法時,其他想要調用此方法的線程就要block,多線程并發量大的時候會對性能有一定的影響。

3.使用ThreadLocal: 

package com.peidasoft.dateformat;import java.text.DateFormat;import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Date;public class ConcurrentDateUtil {    private static ThreadLocalthreadLocal = new ThreadLocal() {        @Override        protected DateFormat initialValue() {            return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");        }    };    public static Date parse(String dateStr) throws ParseException {        return threadLocal.get().parse(dateStr);    }    public static String format(Date date) {        return threadLocal.get().format(date);    }}

另外一種寫法:

package com.peidasoft.dateformat;import java.text.DateFormat;import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Date;public class ThreadLocalDateUtil {    private static final String date_format = "yyyy-MM-dd HH:mm:ss";    private static ThreadLocalthreadLocal = new ThreadLocal();      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);    }   }

說明:使用ThreadLocal, 也是將共享變量變為獨享,線程獨享肯定能比方法獨享在并發環境中能減少不少創建對象的開銷。如果對性能要求比較高的情況下,一般推薦使用這種方法。

4.拋棄JDK,使用其他類庫中的時間格式化類:

1.使用Apache commons 里的FastDateFormat,宣稱是既快又線程安全的SimpleDateFormat, 可惜它只能對日期進行format, 不能對日期串進行解析。

2.使用Joda-Time類庫來處理時間相關問題

做一個簡單的壓力測試,方法一最慢,方法三最快,但是就算是最慢的方法一性能也不差,一般系統方法一和方法二就可以滿足,所以說在這個點很難成為你系統的瓶頸所在。從簡單的角度來說,建議使用方法一或者方法二,如果在必要的時候,追求那么一點性能提升的話,可以考慮用方法三,用ThreadLocal做緩存。

Joda-Time類庫對時間處理方式比較完美,建議使用。

責任編輯:

標簽:

相關推薦:

精彩放送:

新聞聚焦
Top 一区二区三区电影_国产伦精品一区二区三区视频免费_亚洲欧美国产精品va在线观看_国产精品一二三四
正在播放欧美一区| 亚洲国产一区在线| 好男人免费精品视频| 性做久久久久久免费观看欧美| 欧美精品一区二区三区在线看午夜| 在线欧美小视频| 欧美精品一区二区三| 亚洲香蕉在线观看| 国产一区 二区 三区一级| 久久综合狠狠综合久久综青草 | 欧美好骚综合网| av不卡免费看| 国产色婷婷国产综合在线理论片a| 狂野欧美激情性xxxx| 999亚洲国产精| 国产日韩一区二区三区在线播放| 久久一综合视频| 亚洲一区美女视频在线观看免费| 国产主播一区二区| 欧美日韩视频在线| 麻豆成人综合网| 欧美一区二区三区日韩视频| 91久久精品视频| 国产三级精品在线不卡| 欧美日韩免费观看一区| 久久久久久久一区二区| 一区二区三区 在线观看视| 激情欧美国产欧美| 欧美视频中文一区二区三区在线观看| 欧美在线二区| 欧美a一区二区| 午夜精品国产更新| 亚洲主播在线观看| 在线观看亚洲视频啊啊啊啊| 久久久91精品国产一区二区精品| 亚洲高清一区二| 国产私拍一区| 国产精品v亚洲精品v日韩精品| 久久久另类综合| 亚洲专区一区| 亚洲精品资源| 亚洲高清在线播放| 国语自产在线不卡| 国产精品大片wwwwww| 欧美丰满高潮xxxx喷水动漫| 久久精品视频一| 午夜在线观看免费一区| 亚洲一区二区三| 亚洲最新在线视频| 亚洲乱码国产乱码精品精98午夜 | 亚洲一区二区三区成人在线视频精品| 在线观看三级视频欧美| 国内精品久久久久影院优| 国产精品一区二区你懂的| 国产精品v欧美精品∨日韩| 欧美日本在线播放| 欧美连裤袜在线视频| 欧美精品福利在线| 欧美精品久久久久久久| 欧美精品国产| 欧美日韩国产专区| 欧美日韩精品一二三区| 国产精品h在线观看| 国产精品美女久久久久久2018 | 国产精品久久影院| 欧美性猛交xxxx乱大交蜜桃| 欧美午夜视频| 国产精品视频一| 国产一区二区你懂的| 精品av久久707| 亚洲国产美女久久久久| 亚洲精品专区| 亚洲欧美日本伦理| 久久久久欧美| 欧美破处大片在线视频| 欧美日一区二区在线观看 | 一区二区视频免费在线观看 | 黑丝一区二区三区| 亚洲国产美女| 亚洲特黄一级片| 久久久91精品| 欧美精品一二三| 欧美视频国产精品| 国产一区91精品张津瑜| 亚洲黄色在线观看| 亚洲一区二区三区午夜| 久久精品一区二区国产| 欧美国产在线电影| 国产精品色一区二区三区| 激情六月婷婷久久| 亚洲一区二区三区在线| 久久综合影视| 国产精品一区=区| 亚洲人成在线观看网站高清| 午夜精品视频在线| 欧美二区在线| 国模私拍一区二区三区| 亚洲三级免费电影| 久久不见久久见免费视频1| 欧美激情在线观看| 国产日本亚洲高清| 一区二区三区日韩在线观看| 久久久精品日韩| 国产精品国产成人国产三级| 亚洲大胆人体视频| 欧美一区二区| 国产精品久久久久影院亚瑟| 亚洲国产日本| 久久嫩草精品久久久精品| 国产精品久在线观看| 亚洲美女精品成人在线视频| 久久精品99国产精品日本| 国产精品久久久久av| 亚洲精品中文字| 久久在线免费观看视频| 国产精品视频一| 亚洲手机成人高清视频| 欧美精品激情blacked18| 永久免费精品影视网站| 久久国产精品72免费观看| 国产精品久久久一本精品| 日韩午夜视频在线观看| 欧美国产精品| 亚洲国产婷婷香蕉久久久久久99| 久久视频在线视频| 激情视频一区| 久久久欧美精品| 伊人久久大香线| 久久夜色精品国产| 曰韩精品一区二区| 久热精品视频在线观看一区| 极品av少妇一区二区| 久久亚洲欧美| 亚洲国产精品www| 欧美黄色精品| 夜色激情一区二区| 国产精品久久久久久久久免费樱桃 | 欧美日本中文| 日韩午夜一区| 欧美系列一区| 亚洲自拍偷拍色片视频| 国产精品久久一区二区三区| 亚洲视频观看| 国产亚洲欧美日韩日本| 久久裸体艺术| 亚洲精选久久| 国产精品日日摸夜夜添夜夜av| 欧美一区二区三区视频在线| 国产综合色产| 欧美成人黑人xx视频免费观看| 亚洲精品欧美在线| 国产精品swag| 久久久青草婷婷精品综合日韩| 在线观看欧美黄色| 欧美日韩免费视频| 午夜精品一区二区三区在线播放| 黄色欧美日韩| 欧美日韩免费高清一区色橹橹| 亚洲欧美国产高清va在线播| 狠狠色狠狠色综合系列| 欧美精品18| 欧美一区二区三区四区高清| 亚洲第一毛片| 国产精品毛片在线| 欧美sm重口味系列视频在线观看| 一区二区三区视频在线看| 国内成+人亚洲| 欧美日韩理论| 久久久久9999亚洲精品| 一区二区三区不卡视频在线观看 | 亚洲精品国精品久久99热| 欧美图区在线视频| 老司机精品福利视频| 亚洲视频大全| 亚洲国产综合91精品麻豆| 国产农村妇女精品| 欧美日韩国产一级片| 久久精品一区二区三区不卡牛牛 | 91久久精品网| 国产精品专区h在线观看| 欧美激情视频一区二区三区不卡| 亚洲一区国产| 亚洲欧洲日韩在线| 国内精品**久久毛片app| 欧美日韩国产精品一卡| 久久久久一区二区| 性欧美video另类hd性玩具| 夜夜嗨av一区二区三区中文字幕 | 亚洲国产毛片完整版| 国产乱肥老妇国产一区二| 欧美日韩国产成人高清视频| 久久久久久日产精品| 亚洲专区免费| aa级大片欧美| 亚洲精品美女久久7777777| 精品69视频一区二区三区| 国产欧美一区二区精品婷婷| 国产精品xxxav免费视频| 欧美精品在线免费| 欧美激情在线狂野欧美精品|