[Java]Java日期及时间库插件 -- Joda Time.

来到新公司工作也有一个多月了, 陆陆续续做了一些简单的项目. 今天做一个新东西的时候发现了 Joda Time的这个东西, 因为以前用的都是JDK原生的时间处理API, 大家都知道Java原生的时间处理的API一直都是不太好用, 所以这个有必要去学习下, 去总结下.
来到新公司学到的东西挺多的, 比如我们用了Guava, ElasticSearch, kafka/mns/ons, GuavaCache/Ehcache/Memcahe .... 等很多东西都是我以前没有接触过的. 所以待我学习的东西还有很多. 每天都是怀着激动的心情, 去汲取更多的知识. 说了这么多屁话, 开始总结了.

一: 使用示例 
因为双十一就快到了, 所以接触了一个秒杀的活动, 其中有这么一块代码: 

1 /**
2  * 检查抽奖时间
3  *
4  * @return true 可以抽奖 false 不能抽奖
5  */
6  private boolean checkDrawTime() {
7      return DateTime.parse("2016-11-11 11:11:00", DateTimeFormat.forPattern("yyyy-MM-dd HH:mm:ss")).isBeforeNow();
8  }

这个方法很简单, 就是检查现在的时间是不是2016/11/11 11:11:00 之前. 当然这里的DateTime是Jode Time里的类.
查看源码可以知道, DateTime.parse就是将输入的时间转换为后面的格式, 转换的结果为:


然后再用isBeforeNow方法来比较, 下面看下这个方法具体的做法:

1 public boolean isBeforeNow() {
2     return this.isBefore(DateTimeUtils.currentTimeMillis());
3 }
4 
5 public boolean isBefore(long var1) {
6     return this.getMillis() < var1;
7 }

这样看过来是不是很清楚了, 当然Java 自带的JDK也是可以完成的, 但是Joda Time都给我封装好了, 用起来就更加简单了.

接下来看看更多的例子:

1、创建任意时间对象

1 //jdk
2 Calendar calendar=Calendar.getInstance();
3 calendar.set(2012, Calendar.NOVEMBER, 15, 18, 23,55);
4 
5 //Joda-time
6 DateTime dateTime=new DateTime(2012, 12, 15, 18, 23,55);


2、计算两日期相差的天数

 1     //jdk
 2     Calendar start = Calendar.getInstance(); 
 3     start.set(2012, Calendar.NOVEMBER, 14);
 4 
 5     Calendar end = Calendar.getInstance();
 6     end.set(2012, Calendar.NOVEMBER, 15);
 7 
 8     long startTim = start.getTimeInMillis();
 9     long endTim = end.getTimeInMillis();
10     long diff = endTim-startTim;
11     int days=(int) (diff/1000 / 3600 / 24);
12 
13 
14     //joda-time
15     LocalDate start=new LocalDate(2012, 12,14);
16     LocalDate end=new LocalDate(2012, 12, 15);
17     int days = Days.daysBetween(start, end).getDays();

3、获取18天之后的某天在下个月的当前周的第一天日期

 1 //jdk
 2 Calendar current = Calendar.getInstance();
 3 current.add(Calendar.DAY_OF_MONTH, 18);
 4 current.add(Calendar.MONTH, 1);
 5      ......
 6 DateFormat dateFormat=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
 7 Date date = current.getTime();
 8 String dateStr = dateFormat.format(date);
 9 System.out.println(dateStr);
10 
11 //joda-time
12 String dateStr = new DateTime().plusDays(18).plusMonths(1)
13     .dayOfWeek().withMinimumValue().toString("yyyy-MM-dd HH:mm:ss");
14 System.out.println(dateStr);

4、时间格式化

 1 DateTimeFormatter format = DateTimeFormat .forPattern("yyyy-MM-dd HH:mm:ss");
 2 
 3 //时间解析
 4 DateTime dateTime = DateTime.parse("2012-12-21 23:22:45", format);
 5 
 6 //时间格式化,输出==> 2012/12/21 23:22:45 Fri
 7 String string_u = dateTime.toString("yyyy/MM/dd HH:mm:ss EE");
 8 System.out.println(string_u);
 9 
10 //格式化带Locale,输出==> 2012年12月21日 23:22:45 星期五
11 String string_c = dateTime.toString("yyyy年MM月dd日 HH:mm:ss EE",Locale.CHINESE);
12 System.out.println(string_c);

5、与JDK互操作

 1 //通过jdk时间对象构造
 2 Date date = new Date();
 3 DateTime dateTime = new DateTime(date);
 4 
 5 Calendar calendar = Calendar.getInstance();
 6 dateTime = new DateTime(calendar);
 7 
 8 // Joda-time 各种操作.....
 9 dateTime = dateTime.plusDays(1) // 增加天
10     .plusYears(1)// 增加年
11     .plusMonths(1)// 增加月
12     .plusWeeks(1)// 增加星期
13     .minusMillis(1)// 减分钟
14     .minusHours(1)// 减小时
15     .minusSeconds(1);// 减秒数
16     
17 // 计算完转换成jdk 对象
18 Date date2 = dateTime.toDate();
19 Calendar calendar2 = dateTime.toCalendar(Locale.CHINA)


二: api简介
Maven项目组中引入Joda Time坐标地址:

<!-- joda -->
<dependency>
    <groupId>joda-time</groupId>
    <artifactId>joda-time</artifactId>
    <version>2.3</version>
</dependency>

1, 日期和时间:
Joda-time里面的时间全都是不可变的,也就是不可变性。

DateTime :类似于JDK中的Calendar,提供了更多的方法。
DateMidnight :这个概念稍微复杂,封装某个时区(通常为默认时区)在特定年/月/日的午夜时分的时刻。它基本上类似于
DateTime,不同之处在于时间部分总是为与该对象关联的特定
DateTimeZone 时区的午夜时分。
LocalDate :无时间的类,只包含年月日。(不包含时区)
LocalTime :无日期的类,只包含时间。(不包含时区)
LocalDateTime :包含日期和时间。(不包含时区)

下面举一些简单的例子说明Joda-time的强大之处:

 1 //获取当前月份:
 2 DateTime dt = new DateTime();
 3 int month = dt.getMonthOfYear();
 4 //设定2000年/获取当前小时+2的小时:
 5 DateTime dt = new DateTime();
 6 DateTime year2000 = dt.withYear(2000);
 7 DateTime twoHoursLater = dt.plusHours(2);
 8 System.out.println(year2000.getYear());
 9 System.out.println(twoHoursLater.getHourOfDay());
10 //以及下面的一些操作:
11 String monthName = dt.monthOfYear().getAsText();
12 String frenchShortName = dt.monthOfYear().getAsShortText(Locale.FRENCH);
13 boolean isLeapYear = dt.year().isLeap();
14 DateTime rounded = dt.dayOfMonth().roundFloorCopy();
15 System.out.println(monthName); // 本地月份名称
16 System.out.println(frenchShortName); // 法语月份简称
17 System.out.println(isLeapYear); // 是否闰年
18 System.out.println(rounded); // 获取当天0:0:0的DateTime对象

 

2, 日历系统及时区

Joda-time支持多种年表(即日历系统)及所有时区。主要的两个类如下:
Chronology :年表,Joda-time支持很多年表。就像JDK有Calendar,也有GregorianCalendar。
DateTimeZone :时区。

示例如下,一般系统都没有这个需求。

1 Chronology coptic = CopticChronology.getInstance();
2 System.out.println(coptic); // CopticChronology[Asia/Shanghai]
3 
4 DateTimeZone zone = DateTimeZone.forID("Asia/Tokyo");
5 Chronology gregorianJuian = GJChronology.getInstance(zone);
6 System.out.println(gregorianJuian); // GJChronology[Asia/Tokyo]

 

3, 时间跨度

Joda-time支持的时间跨度有三个,分别如下:
Interval :根据开始、结束时间表示一个特定的时间跨度。其为半开区间,即包括起始时刻,不包含结束时刻。
Period :Period设定了一些特定的时间跨度,比如小时、天、月。
Duration :表示绝对的精准跨度,以毫秒为单位。

使用示例如下:

 1 DateTime beginDateTime = new DateTime(2010, 1, 1, 0, 0, 0);
 2 DateTime endDateTime = new DateTime(2015, 1, 1, 0, 0, 0);
 3 Interval interval = new Interval(beginDateTime, endDateTime);
 4 System.out.println(interval.containsNow()); // 是否包含当前时间
 5 System.out.println(interval.toDurationMillis()); // 时间间隔毫秒数
 6  
 7 DateTime dt = new DateTime(2005, 3, 26, 12, 0, 0, 0);
 8 DateTime plusPeriod = dt.plus(Period.days(1));
 9 System.out.println(plusPeriod); // Period的方式+1天
10 DateTime plusDuration = dt.plus(new Duration(24L * 60L * 60L * 1000L));
11 System.out.println(plusDuration); // Duration的方式+1天


三: Joda-time使用的理念

不可变性(Immutability)
瞬间性(Instant)
局部性(Partial)
年表(Chronology)
时区(Time zone)

不可变
Joda-time的类具有不可变性,因此它们的实例无法被修改。(不可变类的一个优点就是它们是线程 安全 的)。用于处理日期计算的 API 方法全部返回一个对应 Joda-time 类的新实例,同时保持原始实例不变。当我们通过一个 API 方法操作 Joda 类时,我们必须捕捉该方法的返回值,因为我们正在处理的实例不能被修改。你可能对这种模式很熟悉,这正是 java.lang.String 的各种操作方法的工作 方式。

瞬间性
Instant 表示时间上的某个精确的时刻,使用从 epoch 开始计算的毫秒表示。这一定义与 JDK 相同,这就是为什么任何 Joda Instant 子类都可以与 JDK Date 和 Calendar 类兼容的原因。
更通用一点的定义是:一个瞬间 就是指时间线上只出现一次且唯一的一个时间点,并且这种日期结构只能以一种有意义的方式出现一次。

局部性
一个局部时间,正如我将在本文中将其称为局部时间片段一样,它指的是时间的一部分片段。瞬间性指定了与 epoch 相对的时间上的一个精确时刻,与此相反,局部时间片段指的是在时间上可以来回 “移动” 的一个时刻,这样它便可以应用于多个实例。比如,6 月 2 日 可以应用于任意一年的 6 月份(使用 Gregorian 日历)的第二天的任意瞬间。同样,11:06 p.m. 可以应用于任意一年的任意一天,并且每天只能使用一次。即使它们没有指定一个时间上的精确时刻,局部时间片段仍然是有用的。
我喜欢将局部时间片段看作一个重复周期中的一点,这样的话,如果我正在考虑的日期构建可以以一种有意义的方式出现多次(即重复的),那么它就是一个局部时间。

年表
Joda 本质——以及其设计核心——的关键就是年表(它的含义由一个同名抽象类捕捉)。从根本上讲,年表是一种日历系统——种计算时间的特殊方式——并且是一种在其中执行日历 算法 的框架。受 Joda 支持的年表的例子包括:ISO(默认)、Coptic、Julian、Islamic等。

时区
时区是值一个相对于英国格林威治的地理位置,用于计算时间。要了解事件发生的精确时间,还必须知道发生此事件的位置。任何严格的时间计算都必须涉及时区(或相对于 GMT),除非在同一个时区内发生了相对时间计算(即时这样时区也很重要,如果事件对于位于另一个时区的各方存在利益关系的话)。
DateTimeZone 是 Joda 库用于封装位置概念的类。许多日期和时间计算都可以在不涉及时区的情况下完成,但是仍然需要了解 DateTimeZone 如何影响 Joda 的操作。默认时间,即从运行代码的机器的系统时钟检索到的时间,在大部分情况下被使用。

好了, 我所了解的就是这么多, 我主要还是从如何使用的角度去学习这个东西, 最后总结了它的几个特性, 当然Joda-Time的强大之处远不止这些, 大家在以后的工作中可以尝试着去使用它. 

 

posted @ 2016-11-06 22:55  一枝花算不算浪漫  阅读(4489)  评论(0编辑  收藏  举报