Java-小技巧-004-jdk时间,jdk8时间,joda,calendar,获取当前时间前一周、前一月、前一年的时间...
1、推薦使用java8 localdate等 線程安全 支持較好
地址
2、joda
一、簡述
查看SampleDateFormat源碼,敘述有:
* 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.Date Formats 非線程安全
建議為每個線程創建單獨的格式實例。
如果多個線程同時訪問一個格式,它必須被同步外部
1、parse()測試
1.1、代碼示例
package com.jd.ofc.trace.bi.util;import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors;/*** @author lihongxu6* @since 2018/1/12 14:36*/ public class DateTest2 extends Thread {private static SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");private String name;private String dateStr;public DateTest2(String name, String dateStr) {this.name = name;this.dateStr = dateStr;}@Overridepublic void run() {Date date = null;try {date = sdf.parse(dateStr);} catch (ParseException e) {e.printStackTrace();}System.out.println(name + " : date: " + date);}public static void main(String[] args) throws InterruptedException {ExecutorService executor = Executors.newCachedThreadPool();executor.execute(new DateTest2("Test_A", "2000-04-28"));executor.execute(new DateTest2("Test_B", "2017-04-28"));executor.execute(new DateTest2("Test_C", "2018-04-28"));executor.shutdown();} } View Code會出現兩種情況:
1>答案不準確
2>代碼異常:
Exception in thread "pool-1-thread-1" Exception in thread "pool-1-thread-2" java.lang.NumberFormatException: multiple pointsat sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1890)at sun.misc.FloatingDecimal.parseDouble(FloatingDecimal.java:110)at java.lang.Double.parseDouble(Double.java:538)at java.text.DigitList.getDouble(DigitList.java:169)at java.text.DecimalFormat.parse(DecimalFormat.java:2056)at java.text.SimpleDateFormat.subParse(SimpleDateFormat.java:1869)at java.text.SimpleDateFormat.parse(SimpleDateFormat.java:1514)at java.text.DateFormat.parse(DateFormat.java:364)at com.jd.ofc.trace.bi.util.DateTest2.run(DateTest2.java:25)at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)at java.lang.Thread.run(Thread.java:745) java.lang.NumberFormatException: multiple pointsat sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1890)at sun.misc.FloatingDecimal.parseDouble(FloatingDecimal.java:110)at java.lang.Double.parseDouble(Double.java:538)at java.text.DigitList.getDouble(DigitList.java:169)at java.text.DecimalFormat.parse(DecimalFormat.java:2056)at java.text.SimpleDateFormat.subParse(SimpleDateFormat.java:1869)at java.text.SimpleDateFormat.parse(SimpleDateFormat.java:1514)at java.text.DateFormat.parse(DateFormat.java:364)at com.jd.ofc.trace.bi.util.DateTest2.run(DateTest2.java:25)at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)at java.lang.Thread.run(Thread.java:745) Test_C : date: Sat Apr 28 00:00:00 CST 2018 View Code2、format測試?
SampleDateFormat源碼format實現
// Called from Format after creating a FieldDelegateprivate StringBuffer format(Date date, StringBuffer toAppendTo,FieldDelegate delegate) {// Convert input date to time field listcalendar.setTime(date);calendar的操作并非是線程安全的,在并發情景下,format的使用并不安全,測試過程與對parse過程的測試相似
二、解決
既然SimpleDateFormat本身并不安全,那么解決的方式無非兩種:優化使用過程或者找替代品。
2.1、臨時創建
不使用Static,每次使用時,創建新實例。
public class DateTest {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);} } View Code存在的問題:
SimpleDateFormat中使用了Calendar對象,由于該對象相當重,在高并發的情況下會大量的new SimpleDateFormat以及銷毀SimpleDateFormat,極其耗費資源。
2.2、synchronized
以synchronized同步SimpleDateFormat對象。
存在的問題:
高并發時,使用該對象會出現阻塞,當前使用者使用時,其他使用者等待,盡管結果是對的,但是并發成了排隊,實際上并沒有解決問題,還會對性能以及效率造成影響。
2.3、ThreadLocal
使用ThreadLocal,令每個線程創建一個當前線程的SimpleDateFormat的實例對象。
存在的問題:
使用ThreadLocal時,如果執行原子任務的過程是每一個線程執行一個任務,那么這樣的聲明基本和每次使用前創建實例對象是沒區別的;如果使用的是多線程加任務隊列,舉個例子,tomcat有m個處理線程,外部有n個待處理任務請求,那么當執行n個任務時,其實只會創建m個SimpleDateFormat實例,對于單一的處理線程,執行任務是有序的,所以對于當前線程而言,不存在并發。
2.4、Apache的 DateFormatUtils 與 FastDateFormat
使用org.apache.commons.lang.time.FastDateFormat 與 org.apache.commons.lang.time.DateFormatUtils。
存在的問題:
apache保證是線程安全的,并且更高效。但是DateFormatUtils與FastDateFormat這兩個類中只有format()方法,所有的format方法只接受long,Date,Calendar類型的輸入,轉換成時間串,目前不存在parse()方法,可由時間字符串轉換為時間對象。
2.5、Joda-Time
使用Joda-Time類庫。
存在的問題:暫無
1、使用maven包
<!-- https://mvnrepository.com/artifact/joda-time/joda-time --> <dependency><groupId>joda-time</groupId><artifactId>joda-time</artifactId><version>2.9.9</version> </dependency>2、使用
Joda-Time — 面向 Java 應用程序的日期/時間庫的替代選擇,Joda-Time 令時間和日期值變得易于管理、操作和理解。事實上,易于使用是 Joda 的主要設計目標。其他目標包括可擴展性、完整的特性集以及對多種日歷系統的支持。并且 Joda 與 JDK 是百分之百可互操作的,因此您無需替換所有 Java 代碼,只需要替換執行日期/時間計算的那部分代碼。
?1.創建一個用時間表示的某個隨意的時刻 — 比如,2015年12月21日0時0分
DateTime dt = new DateTime(2015, 12, 21, 0, 0, 0, 333);// 年,月,日,時,分,秒,毫秒2.格式化時間輸出
DateTime dateTime = new DateTime(2015, 12, 21, 0, 0, 0, 333); System.out.println(dateTime.toString("yyyy/MM/dd HH:mm:ss EE"));3.解析文本格式時間
DateTimeFormatter format = DateTimeFormat .forPattern("yyyy-MM-dd HH:mm:ss"); DateTime dateTime = DateTime.parse("2015-12-21 23:22:45", format); System.out.println(dateTime.toString("yyyy/MM/dd HH:mm:ss EE"));4.在某個日期上加上90天并輸出結果
DateTime dateTime = new DateTime(2016, 1, 1, 0, 0, 0, 0); System.out.println(dateTime.plusDays(90).toString("E MM/dd/yyyy HH:mm:ss.SSS");注意:plus后原值沒變,返回一個新的Datetime
5.到新年還有多少天?
public Days daysToNewYear(LocalDate fromDate) {LocalDate newYear = fromDate.plusYears(1).withDayOfYear(1);return Days.daysBetween(fromDate, newYear); }6.與JDK日期對象的轉換
DateTime dt = new DateTime(); //轉換成java.util.Date對象 Date d1 = new Date(dt.getMillis()); Date d2 = dt.toDate();7.時區
//默認設置為日本時間 DateTimeZone.setDefault(DateTimeZone.forID("Asia/Tokyo")); DateTime dt1 = new DateTime(); System.out.println(dt1.toString("yyyy-MM-dd HH:mm:ss"));//倫敦時間 DateTime dt2 = new DateTime(DateTimeZone.forID("Europe/London")); System.out.println(dt2.toString("yyyy-MM-dd HH:mm:ss"));8.計算間隔和區間
DateTime begin = new DateTime("2015-02-01"); DateTime end = new DateTime("2016-05-01"); //計算區間毫秒數 Duration d = new Duration(begin, end); long millis = d.getMillis(); //計算區間天數 Period p = new Period(begin, end, PeriodType.days()); int days = p.getDays(); //計算特定日期是否在該區間內 Interval interval = new Interval(begin, end); boolean contained = interval.contains(new DateTime("2015-03-01"));9.日期比較
DateTime d1 = new DateTime("2015-10-01"); DateTime d2 = new DateTime("2016-02-01"); //和系統時間比 boolean b1 = d1.isAfterNow(); boolean b2 = d1.isBeforeNow(); boolean b3 = d1.isEqualNow(); //和其他日期比 boolean f1 = d1.isAfter(d2); boolean f2 = d1.isBefore(d2); boolean f3 = d1.isEqual(d2);資料:
Joda-Time 簡介(中文)https://www.ibm.com/developerworks/cn/java/j-jodatime.html
Joda-Time 文檔(英文)http://joda-time.sourceforge.net/
3、傳統方式
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");Calendar c = Calendar.getInstance();//過去七天c.setTime(new Date());c.add(Calendar.DATE, - 7);Date d = c.getTime();String day = format.format(d);System.out.println("過去七天:"+day);//過去一月c.setTime(new Date());c.add(Calendar.MONTH, -1);Date m = c.getTime();String mon = format.format(m);System.out.println("過去一個月:"+mon);//過去三個月c.setTime(new Date());c.add(Calendar.MONTH, -3);Date m3 = c.getTime();String mon3 = format.format(m3);System.out.println("過去三個月:"+mon3);//過去一年c.setTime(new Date());c.add(Calendar.YEAR, -1);Date y = c.getTime();String year = format.format(y);System.out.println("過去一年:"+year);?
總結
以上是生活随笔為你收集整理的Java-小技巧-004-jdk时间,jdk8时间,joda,calendar,获取当前时间前一周、前一月、前一年的时间...的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 分辨真假数据科学家的20个问题及回答
- 下一篇: JVM基本原理