Java 8 的日期和时间API

Java 8 以前的日期和时间 API 设计有些不足,Java 8 引入了一套新的API,位于 java.time 包下。

Instant

Instant 表示时刻,不直接对应年月日信息,需要通过时区转换。

Instant now = Instant.now();

可以根据 Epoch Time(纪元时)创建 Instant。

Instant now = Instant.ofEpochMilli(System.currentTimeMillis());

Instant 和 Date 可以通过纪元时相互转换:

  • Date 转 Instant
public static Instant toInstant(Date date) {
    return Instant.ofEpochMilli(date.getTime());
}
  • Instant 转 Date
public static Date toDate(Instant instant) {
    return new Date(instant.toEpochMilli());
}     

给定一个时刻,使用不同时区解读,日历信息是不同的,默认时区是 ZoneId.systemDefault()。

public ZonedDateTime atZone(ZoneId zone)

LocalDateTime

LocalDateTime 表示与时区无关的日期和时间,不直接对应时刻,需要通过时区转换。

获取系统默认时区的当前日期和时间:

LocalDateTime dateTime = LocalDateTime.now();

直接用年月日等信息构建 LocalDateTime:

LocalDateTime ldt = LocalDateTime.of(2017, 7, 11, 20, 45, 5);

获取日历信息:

public int getYear()

public int getMonthValue()

public int getDayOfMonth()

public int getHour()

public int getMinute()

public int getSecond()

public DayOfWeek getDayOfWeek()  

DayOfWeek 是一个枚举,有 7 个取值,从 DayOfWeek.MONDAY 到 DayOfWeek.SUN-DAY。

ZoneOffset

LocalDateTime 不能直接转为时刻 Instant,转换需要一个参数 ZoneOffset。

ZoneOffset 表示相对于格林尼治的时区差,北京是 +08:00。

public static Instant toBeijingInstant(LocalDateTime ldt) {
  return ldt.toInstant(ZoneOffset.of("+08:00"));
}

LocalDate/LocalTime

可以认为 LocalDateTime 由两部分组成,一部分是日期 LocalDate,另一部分是时间 LocalTime。

// 表示2017年7月11日
LocalDate ld = LocalDate.of(2017, 7, 11);
// 当前时刻按系统默认时区解读的日期
LocalDate now = LocalDate.now();

// 表示21点10分34秒
LocalTime lt = LocalTime.of(21, 10, 34);
// 当前时刻按系统默认时区解读的时间
LocalTime time = LocalTime.now();
LocalDateTime ldt = LocalDateTime.of(2017, 7, 11, 20, 45, 5);

LocalDate ld = ldt.toLocalDate(); // 2017-07-11
LocalTime lt = ldt.toLocalTime(); // 20:45:05

// LocalDate加上时间,结果为2017-07-11 21:18:39
LocalDateTime ldt2 = ld.atTime(21, 18, 39);

// LocalTime加上日期,结果为2016-03-24 20:45:05
LocalDateTime ldt3 = lt.atDate(LocalDate.of(2016, 3, 24));

ZonedDateTime

ZonedDateTime 表示特定时区的日期和时间,获取系统默认时区的当前日期和时间,除了记录日历信息,还会记录时区。

LocalDateTime.now() 也是获取默认时区的当前日期和时间,但 LocalDateTime 只单纯记录年月日时分秒等信息。

// 根据Instant和时区构建ZonedDateTime
public static ZonedDateTime ofInstant(Instant instant, ZoneId zone)
  
// 根据LocalDate、LocalTime和ZoneId构造
public static ZonedDateTime of(LocalDate date, LocalTime time, ZoneId zone)

ZonedDateTime 可以直接转换为 Instant。

ZonedDateTime ldt = ZonedDateTime.now();
Instant now = ldt.toInstant();

DateTimeFormatter 格式化

java.time.format.DateTimeFormatter 是线程安全的,用法和 DateFormat 相差不大。

  • DateTimeFormatter.ofPattern(pattern) 实例化
  • format 格式化 TemporalAccessor 接口子类对象转字符串
  • parse 按格式解析字符串转为 TemporalAccessor 接口子类对象
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
LocalDateTime dateTime = LocalDateTime.now();
System.out.println(formatter.format(dateTime));
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
String str = "2016-08-18 14:20:45";
LocalDateTime ldt = LocalDateTime.parse(str, formatter);

设置和修改时间

修改时期和时间有两种方式,一种是直接设置绝对值,另一种是在现有值的基础上进行相对增减操作。

Java 8的大部分日期和时间类都是不可变类,修改操作是通过创建并返回新对象来实现的,原对象本身不会变。

  • 直接设置

设置时间为下午 3 点 20 分:

LocalDateTime ldt = LocalDateTime.now();
ldt = ldt.withHour(15).withMinute(20).withSecond(0).withNano(0);
LocalDateTime ldt = LocalDateTime.now();
ldt = ldt.toLocalDate().atTime(15, 20);

设置时间为 0 点:

LocalDateTime ldt = LocalDateTime.now();
ldt = ldt.with(ChronoField.MILLI_OF_DAY, 0); // ChronoField 是一个枚举
LocalDateTime ldt = LocalDateTime.of(LocalDate.now(), LocalTime.MIN);
LocalDateTime ldt = LocalDate.now().atTime(0, 0);
  • 增减操作

设置时间为 3 小时 5 分钟后:

LocalDateTime ldt = LocalDateTime.now();
ldt = ldt.plusHours(3).plusMinutes(5);

设置时间为下一个周二上午 10 点 :

LocalDate ld = LocalDate.now();
if(! ld.getDayOfWeek().equals(DayOfWeek.MONDAY)){
  ld = ld.plusWeeks(1);
}
LocalDateTime ldt = ld.with(ChronoField.DAY_OF_WEEK, 2).atTime(10, 0);

Java 8 有一个专门的接口 TemporalAdjuster,这是一个函数式接口,对日期或时间进行调整。

Instant、LocalDateTime 和 LocalDate 等都实现了它,有一个专门的默认实现类 TemporalAdjusters。

public interface TemporalAdjuster {
  Temporal adjustInto(Temporal temporal);
}

设置时间为下一个周二上午 10 点 :

LocalDate ld = LocalDate.now();
LocalDateTime ldt = ld.with(TemporalAdjusters.next(DayOfWeek.TUESDAY)).atTime(10, 0);

TemporalAdjusters 的 next 实现:

public static TemporalAdjuster next(DayOfWeek dayOfWeek) {
  int dowValue = dayOfWeek.getValue();
  return (temporal) -> {
    int calDow = temporal.get(DAY_OF_WEEK);
    int daysDiff = calDow - dowValue;
    return temporal.plus(daysDiff >= 0 ? 7 - daysDiff : -daysDiff, DAYS);
  };
}

时间段的计算

Java 8 中表示时间段的类主要有两个:Period 和 Duration。

Period 表示日期之间的差,用年月日表示,不能表示时间;

Duration 表示时间差,用时分秒等表示,也可以用天表示,一天严格等于24小时,不能用年月表示。

LocalDate ld1 = LocalDate.of(2016, 3, 24);
LocalDate ld2 = LocalDate.of(2017, 7, 12);
Period period = Period.between(ld1, ld2);
System.out.println(period.getYears() + "年" + period.getMonths() + "月" + period.getDays() + "天");
long lateMinutes = Duration.between(LocalTime.of(9, 0), LocalTime.now()).toMinutes();

与 Date/Calendar 转换

Date 可以与 Instant 通过毫秒数相互转换,对于其他类型,可以通过毫秒数与 Instant 相互转换。

LocalDateTime 转换为 Date:

public static Date toDate(LocalDateTime ldt){
  return new Date(ldt.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli());
}

Date 按默认时区转换为 LocalDateTime:

public static LocalDateTime toLocalDateTime(Date date) {
  return LocalDateTime.ofInstant(Instant.ofEpochMilli(date.getTime()), ZoneId.systemDefault());
}

ZonedDateTime 转换为 Calendar:

public static Calendar toCalendar(ZonedDateTime zdt) {
  TimeZone tz = TimeZone.getTimeZone(zdt.getZone());
  Calendar calendar = Calendar.getInstance(tz);
  calendar.setTimeInMillis(zdt.toInstant().toEpochMilli());
  return calendar;
}

Calendar 转换为 ZonedDateTime:

public static ZonedDateTime toZonedDateTime(Calendar calendar) {
  ZonedDateTime zdt = ZonedDateTime.ofInstant(
    Instant.ofEpochMilli(calendar.getTimeInMillis()),
    calendar.getTimeZone().toZoneId());
  return zdt;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容