91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

simpledateformat線程不安全的原因

發布時間:2021-05-10 15:04:05 來源:億速云 閱讀:366 作者:小新 欄目:編程語言

這篇文章將為大家詳細講解有關simpledateformat線程不安全的原因,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

原因:在多線程環境下,當多個線程同時使用相同的SimpleDateFormat對象(如static修飾)的話,如調用format方法時,多個線程會同時調用calender.setTime方法,導致time被別的線程修改,因此線程是不安全的。

本教程操作環境:windows7系統、java8版、DELL G3電腦。

線程不安全驗證:

/**
 * SimpleDateFormat線程安全測試
 * 〈功能詳細描述〉
 *
 * @author 17090889
 * @see [相關類/方法](可選)
 * @since [產品/模塊版本] (可選)
 */
public class SimpleDateFormatTest {
    private SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    ThreadPoolExecutor poolExecutor = new ThreadPoolExecutor(10, 100, 1, TimeUnit.MINUTES, new LinkedBlockingQueue<>(1000), new MyThreadFactory("SimpleDateFormatTest"));

    public void test() {
        while (true) {
            poolExecutor.execute(new Runnable() {
                @Override
                public void run() {
                    String dateString = simpleDateFormat.format(new Date());
                    try {
                        Date parseDate = simpleDateFormat.parse(dateString);
                        String dateString2 = simpleDateFormat.format(parseDate);
                        System.out.println(dateString.equals(dateString2));
                    } catch (ParseException e) {
                        e.printStackTrace();
                    }
                }
            });
        }
    }

輸出:

  true
  false
  true
  true
  false

出現了false,說明線程不安全

1、format方法

public StringBuffer format(Date date, StringBuffer toAppendTo,
                               FieldPosition pos)
    {
        pos.beginIndex = pos.endIndex = 0;
        return format(date, toAppendTo, pos.getFieldDelegate());
    }

    // Called from Format after creating a FieldDelegate
    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;
    }
 protected Calendar calendar;

可以看到,多個線程之間共享變量calendar,并修改calendar。因此在多線程環境下,當多個線程同時使用相同的SimpleDateFormat對象(如static修飾)的話,如調用format方法時,多個線程會同時調用calender.setTime方法,導致time被別的線程修改,因此線程是不安全的。

此外,parse方法也是線程不安全的,parse方法實際調用的是CalenderBuilder的establish來進行解析,其方法中主要步驟不是原子操作。

解決方案:

  1、將SimpleDateFormat定義成局部變量

  2、 加一把線程同步鎖:synchronized(lock)

  3、使用ThreadLocal,每個線程都擁有自己的SimpleDateFormat對象副本。如:

/**
 * SimpleDateFormat線程安全測試
 * 〈功能詳細描述〉
 *
 * @author 17090889
 * @see [相關類/方法](可選)
 * @since [產品/模塊版本] (可選)
 */
public class SimpleDateFormatTest {
        private static final ThreadLocal<SimpleDateFormat> THREAD_LOCAL = new ThreadLocal<SimpleDateFormat>() {
        @Override
        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        }
    };
    //    private SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    ThreadPoolExecutor poolExecutor = new ThreadPoolExecutor(10, 100, 1, TimeUnit.MINUTES, new LinkedBlockingQueue<>(1000), new MyThreadFactory("SimpleDateFormatTest"));

    public void test() {
        while (true) {
            poolExecutor.execute(new Runnable() {
                @Override
                public void run() {
                    SimpleDateFormat simpleDateFormat = THREAD_LOCAL.get();
                    if (simpleDateFormat == null) {
                        simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                    }
                    String dateString = simpleDateFormat.format(new Date());
                    try {
                        Date parseDate = simpleDateFormat.parse(dateString);
                        String dateString2 = simpleDateFormat.format(parseDate);
                        System.out.println(dateString.equals(dateString2));
                    } catch (ParseException e) {
                        e.printStackTrace();
                    } finally {
                        local.remove();
                    }
                }
            });
        }
    }
}

  4、使用DateTimeFormatter代替SimpleDateFormat

  DateTimeFormatter是線程安全的,默認提供了很多格式化方法,也可以通過ofPattern方法創建自定義格式化方法。

  (1)格式化日期示例:

 LocalDateTime localDateTime = LocalDateTime.now();
 System.out.println(localDateTime); // 2019-11-20T15:04:29.017
 DateTimeFormatter dtf=DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss");
 String strDate=localDateTime.format(dtf);
 System.out.println(strDate); // 2019/23/20 15:23:46

  (2)解析日期

 DateTimeFormatter dtf=DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss");
 LocalDateTime localDateTime=LocalDateTime.parse("2019/11/20 15:23:46",dtf);
 System.out.println(localDateTime); // 2019-11-20T15:23:46

關于“simpledateformat線程不安全的原因”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

湖南省| 河东区| 贺兰县| 密云县| 大厂| 垫江县| 册亨县| 定结县| 瑞丽市| 仙桃市| 秦皇岛市| 平定县| 泸州市| 怀远县| 于田县| 鄂伦春自治旗| 江都市| 德江县| 保定市| 鄱阳县| 固原市| 镇康县| 新龙县| 丰宁| 和政县| 买车| 桓仁| 陇川县| 赫章县| 桑植县| 邹城市| 嘉鱼县| 宁城县| 枣强县| 太康县| 诏安县| 巴林左旗| 武威市| 通山县| 特克斯县| 宜城市|