首页 > 技术文章 > 20145231第七周学习笔记

xzh20145231 2016-04-16 16:08 原文

20145231 《Java程序设计》第7周学习总结

教材学习内容总结

第十三章 时间与日期

认识时间与日期

•就目前来说,即使标注为GMT(无论是文件说明,或者是API的日期时间字符串描述),实际上谈到时间指的是UTC时间。

•秒的单位定义是基于TAI,也就是铯原子辐射振动次数。

•epoch为某个特定时代的开始,时间轴上某一瞬间。

•UTC考虑了地球自转越来越慢而有闰秒修正,确保UTC与UT相差不会超过0.9秒。

•Unix时间是1970年1月1日00:00:00为起点而经过的秒数,不考虑闰秒。

认识Date与Calendar

•如果想要取得系统时间,方法之一是使用System.currentTimeMillis()方法,返回的是long类型整数。如:

import java.util.*;
import static java.lang.System.*;

public class DateDemo {
    public static void main(String[] args) {
        Date date1 = new Date(currentTimeMillis());
        Date date2 = new Date();

        out.println(date1.getTime());
        out.println(date2.getTime());
    }
}

Date有两个构造函数可以使用,一个可使用epoch毫秒数构建,另一个为无自变量构造函数,内部亦是使用System.currentTimeMillis()取得毫秒数,调用getTime()可取得内部保存的epoch毫秒数值。范例执行结果如下:

•DateFormat是个抽象类,其操作类是java.text.SimpleDateFormat,你可以直接构建SimpleDateFormat实例,或是使用DateFormat的getDateInstance()、getTimeInstance()、getDateTimeInstance等静态方法,用较简便方式按不同需求取得SimpleDateFormat实例。

•直接构建SimpleDateFormat的好处是,可使用模式字符串自定义格式。

•SimpleDateFormat还有个parse()方法,可以按构建SimpleDateFormat时指定的格式,将指定的字符串剖析为Date实例。如:

import java.util.*;
import java.text.*;

public class HowOld {
    public static void main(String[] args) throws Exception {
        System.out.print("输入出生年月日(yyyy-mm-dd):");
        DateFormat dateFormat = new SimpleDateFormat("yyyy-mm-dd");
        Date birthDate = dateFormat.parse(new Scanner(System.in).nextLine());
        Date currentDate = new Date();
        long life = currentDate.getTime() - birthDate.getTime();
        System.out.println("你今年的岁数为:" + 
                  (life / (365 * 24 * 60 * 60 * 1000L)));
    }
}


运行结果如图:

•Date现在建议作为时间轴上的瞬间代表,要格式化时间日期则通过DateFormat,如果想要取得某个时间日期信息,或者是对时间日期进行操作,可以使用Calendar实例。如:

Calendar calendar = Calendar.getInstance();

•Calendar是个抽象类,java.util.GregorianCalendar是其子类,操作了儒略历与格里高利历的混合历,通过Calendar的getInstance()取得的Calendar实例,默认就是取得GregorianCalendar实例。如:

out.println(calendar.get(Calendar.YEAR));

out.println(calendar.get(Calendar.MONTH));

out.println(calendar.get(Calendar.DATE));

•若想取得默认时区信息,可以使用java.util.TimeZone的getDefault()方法。如:

import static java.lang.System.out;

import java.util.TimeZone;

示例代码如下:

public class TimeZoneDemo {
    public static void main(String[] args) {
        TimeZone timeZone = TimeZone.getDefault();
        out.println(timeZone.getDisplayName());
        out.println("\t时区ID:" + timeZone.getID());
        out.println("\t日光节约秒数:" + timeZone.getDSTSavings());
        out.println("\tUTC 偏移毫秒数:" + timeZone.getRawOffset());
    }
}

运行结果如图:

JDK8新时间日期API

•可以使用Instance的静态方法now()取得代表Java epoch毫秒数。在取得Instance实例后,可以使用plusSeconds()、plusMills()、plusNanos()、minusSeconds()、minusMills()、minusNanos()来做时间轴上的运算,Instance实例本身不会变动,这些操作都会返回新的Instance实例,代表运算后的瞬时。

•如果取得了Date实例,而想要改用Instance,则可以调用Date市里的toInstance()方法来取得,如果有个Instance实例,可以使用Date的静态方法from()转为Date。

•LocalDateTime、LocalDate、LocalTime等类名称开头为Local,表示它们都只是对时间的描述,并没有时区信息。

•对于时间计量,新时间与日期API以类Duration来定义,可用于计量天、时、分、秒的时间差,精度调整可以达纳秒等级,而秒的最大值可以是long类型保存值。对于年、月、星期、日的时间差,则使用Period类定义。如:

import java.time.*;

import java.util.Scanner;

import static java.lang.System.out;

示例代码如下:


public class HowOld2 {
    public static void main(String[] args) {
        out.print("输入出生年月日(yyyy-mm-dd):");
        LocalDate birth = LocalDate.parse(new Scanner(System.in).nextLine());
        LocalDate now = LocalDate.now();
        Period period = Period.between(birth, now);
        out.printf("你活了 %d 年 %d 月 %d 日%n", 
                period.getYears(), period.getMonths(), period.getDays());
    }
}

运行结果如图:

教材学习中的问题和解决过程

问题:对于Date实例的理解存在疑惑

解决过程:通过仔细阅读书上p437~438内容,理解了之所以Date实例让我困惑的原因是我没有理解或注意机器时间观点和人类时间观点的差异,Date实例真正代表的并不是日期,最接近的概念应该是时间轴上特定的一瞬间,时间精度是毫秒,也就是UTC时间1970年1月1日0时0分0毫秒至某个特定瞬时的毫秒差,代表的是机器的时间概念,而混淆机器与人类时间观点会产生的问题如:日光节约问题,Date实例内部真正可靠的信息只有内含的epoch秒数,若我们取得Date实例时,要获取时间信息应该是通过Date的getTime()取得epoch毫秒数,这样就不会混淆。

代码调试中的问题和解决过程

问题:不太理解LocalDate.of()LocalDateTime.of这几个方法的差别

解决过程:通过运行相关代码有所理解,LocalDate、LocalDateTime、LocalTime这几个类名称开头为Local,表示他们都只是对时间的描述,并没有时区信息。对于LocalDate,设定不存在日期,会抛出DateTimeException,如:LocalDate.of(2014,2,29),因为2014不是闰年,对于LocalDateTime.of(1975,4,1,0,0,0),由于没有时区信息,无法判断该时间是否存在,就不会抛出DateTimeException。

package cc.openhome;

import static java.lang.System.out;
import java.time.*;

public class ZonedDateTimeDemo {
    public static void main(String[] args) {
        LocalTime localTime = LocalTime.of(0, 0, 0);
        LocalDate localDate = LocalDate.of(1975, 4, 1);
        ZonedDateTime zonedDateTime = ZonedDateTime.of(
                localDate, localTime, ZoneId.of("Asia/Taipei"));
        
        out.println(zonedDateTime);
        out.println(zonedDateTime.toEpochSecond());
        out.println(zonedDateTime.toInstant().toEpochMilli());
    }
}

运行结果如图:

其他(感悟、思考等,可选)

本周的内容相对前几周较少,所以相对理解把握的较好,关键在于新的时间API的理解和调用,难点在于把握好人类时间概念与机器时间概念的异同。

代码托管截图

学习进度条

代码行数(新增/累积) 博客量(新增/累积) 学习时间(新增/累积) 重要成长
目标 5000行 30篇 400小时
第一周 200/200 2/2 20/20
第二周 300/500 2/4 21/41
第三周 450/950 3/7 22/63
第四周 450/1365 2/9 20/83
第五周 450/1815 2/11 20/113
第六周 450/2315 2/13 20/133
第七周 300/2615 2/15 15/148 了解了时间与日期相关API

推荐阅读