Java - 日期和时间操作

一、获取当前时间的时间戳

1)时间进制

1秒 = 1000毫秒

1秒 = 1000000微秒(1毫秒=1000微秒)

1秒 = 1000000000纳秒(1微秒=1000纳秒)( 1毫秒=1000000纳秒)

要获得秒级时间戳,可以使用毫秒级时间戳除以1000即可

2)获取毫秒时间戳(13位) - 单位是毫秒

获取毫秒时间戳的方式比较多,一般都是用System.currentTimeMillis()

// 方式1
long timeStamp1 = System.currentTimeMillis();

// 方式2
Date date = new Date();
long timeStamp2 = date.getTime();

// 方式3
Calendar calendar = Calendar.getInstance();
long timeStamp3 = calendar.getTimeInMillis();

// 方法4
Clock clock = Clock.systemUTC();
long timeStamp4 = clock.millis();

// 打印时间戳示例
System.out.println(timeStamp1 + " - " + timeStamp2 + " - " + timeStamp3 + " - " + timeStamp4);

3)获取纳秒时间戳

纳秒时间戳,好像用的不是特别多

System.out.println(System.nanoTime());

二、java.util包

1)Date - 不建议使用

之所以使用java.util.Date指明Date类的包为java.util,是因为java.sql包中也有一个Date类。

Date类的输出格式:Sun Sep 08 17:49:50 CST 2019

Date类有很多方法都被废弃了,包括构造方法,所以常用的使用方法如下:

// 利用当前时间戳创建的Date实例,底层调用System.currentTimeMillis()
Date date1 = new Date();

// 利用一个时间戳来创建Date对象(时间戳转Date对象)
Date date2 = new Date(System.currentTimeMillis());

// Date的compareTo,用于比较两个时间的先后
int flag = date1.compareTo(date2);
// date1.compareTo(date2)返回值分三种情况
// flag = 0, 两个时间相同(时间戳相同)
// flag = 1, date1 要晚于 date2(date1的时间戳大于date2的时间戳)
// flag = -1,date1 要早于 date2(date1的时间戳小于于date2的时间戳)

// 判断date1是否晚于date2
boolean after = date1.after(date2);

// 判断date1是否早于date2
boolean before = date1.before(date2);

// 获取date对象的时间戳(13位),毫秒
long timeStamp = date1.getTime();

// 设置date对象的时间戳
date1.setTime(timeStamp);

建议不要使用Date类的废弃的API,有一个Date.getMonth()获取月份,是从0开始计数,也就是说,3月份,getMonth()的值是2。

2)Calendar

Calendar是一个抽象类,所以不能实例化,但是可以调用getInstance()静态方法获得Calendar实例。

// 调用静态方法获取Calendar实例(使用默认的TimeZone和Locale)
Calendar calendar1 = Calendar.getInstance();

// 获取时间戳(13位),毫秒
long timeStamp = calendar1.getTimeInMillis();

// 获取Date对象
Date date1 = calendar1.getTime();

// 设置时间
Date date2 = new Date(System.currentTimeMillis() - 100);
calendar1.setTime(date2);

// 获取Calendar的字段值(比如YEAR、MONTH....)
final int month = calendar1.get(Calendar.MONTH);
System.out.println(month); // 注意,month索引为0 - 11

// 修改时间(为指定字段进行计算)
calendar1.add(Calendar.YEAR, -1); // 当前是2020年,这里将年份减1
System.out.println(calendar1.get(Calendar.YEAR)); // 2019
// 其他字段,比如月、天、时分秒的计算都是一样的做法

// 获取某个字段的最大值(同样可以有对应的接口获取最小值)
int maxDay = calendar1.getActualMaximum(Calendar.DAY_OF_MONTH);
System.out.println(maxDay); // 2019年3月,最多有31天

Calendar calendar2 = Calendar.getInstance();

// 设置指定字段值,注意,月份是从0开始计数
calendar2.set(Calendar.YEAR, 2030);
System.out.println(calendar2.getTime());// Wed Mar 20 10:27:37 CST 2030

// 设置年月日
calendar2.set(2029, 11, 5);
System.out.println(calendar2.getTime());// Wed Dec 05 10:27:53 CST 2029

// 设置年月日时分秒
calendar2.set(2030, 10, 4, 12, 58, 59);
System.out.println(calendar2.getTime());// Mon Nov 04 12:58:59 CST 2030

三、java.time - 推荐使用

java8新的时间API的使用方式,包括创建、格式化、解析、计算、修改。

Java的Date,Calendar类型使用起来并不是很方便,而且Date类有着线程不安全等诸多弊端。同时若不进行封装,会在每次使用时特别麻烦。

于是Java8推出了线程安全、简易、高可靠的时间包。并且数据库中也支持LocalDateTime类型,在数据存储时候使时间变得简单。

java.time包括三个相关的时间类型:LocalDateTime - 年月日时分秒;LocalDate - 日期;LocalTime - 时间;

1)LocalDate

LocalDate可以说使用的比较多了,因为可以比较方便的获取、设置、修改日期,需要注意的是,LocalDate,从名称上就能看出,这是获取“本地”日期。

// 创建LocalDate对象
LocalDate localDate = LocalDate.now();
System.out.println(localDate); // 2020-03-20

final LocalDate localDate2 = localDate.minusDays(5);
System.out.println(localDate); // 2020-03-20 注意,并不会直接修改LocalDate对象
System.out.println(localDate2); // 2020-03-15

// getXxxx获取年月日信息
final int dayOfMonth = localDate.getDayOfMonth();
System.out.println(dayOfMonth);// getDayOfMonth = 20日

// 设置时间为2019-11-20,月份从1开始
LocalDate localDate3 = LocalDate.of(2019, 9, 14);
System.out.println(localDate3); // 2019-09-14

// 2019年的第100天
LocalDate localDate4 = LocalDate.ofYearDay(2020, 100);
System.out.println(localDate4); // 2020-04-09

// 可以调用LocalData的minusXxx、plusXxx进行日期的计算,getXxx获取某项值
System.out.println(localDate3.plusYears(1));

2)LocalTime

LocalTime,主要是对Time,也就是对时间的操作,并且是本地的时间

// 利用当前时间,创建LocalTime对象
LocalTime localTime1 = LocalTime.now();
System.out.println(localTime1); // 10:44:43.379

// 指定时-分,创建LocalTime对象,注意,小时范围为0-23
LocalTime localTime2 = LocalTime.of(23, 59);
System.out.println(localTime2); // 23:59 注意,输出没有秒数

// 指定时-分-秒,创建LocalTime对象
LocalTime localTime3 = LocalTime.of(12, 35, 40);
System.out.println(localTime3); // 12:35:40

// 额外指定纳秒
LocalTime localTime4 = LocalTime.of(13, 20, 55, 1000);
System.out.println(localTime4);

// 指定一天中的第1000秒来创建LocalTime对象
LocalTime localTime5 = LocalTime.ofSecondOfDay(1000);
System.out.println(localTime5); // 00:16:40

// 利用一天中的纳秒数来创建LocalTime对象
LocalTime localTime6 = LocalTime.ofNanoOfDay(100000000);
System.out.println(localTime6); // 00:00:00.100

// 可以调用LocalTime的plusXxx,minusXxx进行时间计算,getXxx获取某项值
System.out.println(localTime2.plusMinutes(10));// 23:59+10分钟 = 00:09

3)LocalDateTime

LocalDateTime其实就是LocalDate和LocalTime加在一起的类了,使用方式也是一样的:

LocalDateTime now = LocalDateTime.now();
System.out.println(now); // 2020-03-20T10:40:33.204

// 利用LocalDate和LocalTime创建LocalDateTime实例
LocalDateTime localDateTime1 = LocalDateTime.of(LocalDate.now(), LocalTime.now());
System.out.println(localDateTime1); // 2020-03-20T10:40:33.205

// of用法和LocalDate和LocalTime的of用法一样,综合在一起了而已
LocalDateTime localDateTime2 = LocalDateTime.of(2019, 9, 14, 16, 30, 45);
System.out.println(localDateTime2); // 2019-09-14T16:30:45

// 可以调用LocalDateTime的minusXxx和plusXxx来进行日期和时间的操作,使用getXxx获取某个项的值
LocalDateTime localDateTime3 = localDateTime2.plusYears(1);// Year+1
System.out.println(localDateTime3);// 2020-09-14T16:30:45

4)Clock

Clock - 时钟,用法如下:

Clock clock = Clock.systemUTC();

// 获取时区
final ZoneId zone = clock.getZone();
System.out.println(zone); // Z

// 获取时间戳(13位),毫秒
long timeStamp = clock.millis();
System.out.println(timeStamp); // 1584671920052

四、日期时间格式

1)SimpleDateFormat

Date date = new Date();
System.out.println(date); // Fri Mar 20 11:03:41 CST 2020

// 创建想要显示的格式
SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
final String dateStr = formatter.format(date);
System.out.println(dateStr); // 2020-03-20 11:03:41

使用SimpleDateFormat对时间进行格式化,但SimpleDateFormat是线程不安全的 SimpleDateFormat的format方法最终调用代码:

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;
}
View Code

calendar是共享变量,并且这个共享变量没有做线程安全控制。当多个线程同时使用相同的SimpleDateFormat对象【如用static修饰的SimpleDateFormat】调用format方法时,多个线程会同时调用calendar.setTime方法,可能一个线程刚设置好time值另外的一个线程马上把设置的time值给修改了导致返回的格式化时间可能是错误的。

在多并发情况下使用SimpleDateFormat需格外注意:

SimpleDateFormat除了format是线程不安全以外,parse方法也是线程不安全的。parse方法实际调用alb.establish(calendar).getTime()方法来解析,alb.establish(calendar)方法里主要完成了

● 重置日期对象cal的属性值

● 使用calb中中属性设置cal

● 返回设置好的cal对象

但是这三步不是原子操作

多线程并发如何保证线程安全

● 避免线程之间共享一个SimpleDateFormat对象,每个线程使用时都创建一次SimpleDateFormat对象 -> 创建和销毁对象的开销大

● 对使用format和parse方法的地方进行加锁 -> 线程阻塞性能差

● 使用ThreadLocal保证每个线程最多只创建一次SimpleDateFormat对象 -> 较好的方法

Date对时间处理比较麻烦,比如想获取某年、某月、某星期,以及n天以后的时间,如果用Date来处理的话真是太难了,你可能会说Date类不是有getYear、getMonth这些方法吗,获取年月日很Easy,但都被弃用了。

2)DateTimeFormatter ★推荐

LocalDate localDate = LocalDate.of(2019, 9, 14);

// 20190914
String s1 = localDate.format(DateTimeFormatter.BASIC_ISO_DATE);

// 2019-09-14
String s2 = localDate.format(DateTimeFormatter.ISO_LOCAL_DATE);

// 自定义格式化 - 14/09/2019
DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("dd/MM/yyyy");
String s3 = localDate.format(dateTimeFormatter);

DateTimeFormatter默认提供了多种格式化方式,如果默认提供的不能满足要求,可以通过DateTimeFormatter的ofPattern方法创建自定义格式化方式。

解析时间

LocalDate localDate1 = LocalDate.parse("20190914", DateTimeFormatter.BASIC_ISO_DATE);
System.out.println(localDate1);// 2019-09-14

LocalDate localDate2 = LocalDate.parse("2019-09-14", DateTimeFormatter.ISO_LOCAL_DATE);
System.out.println(localDate2);// 2019-09-14

和SimpleDateFormat相比,DateTimeFormatter是线程安全的

posted @ 2020-03-20 11:30  ohmok  阅读(651)  评论(0)    收藏  举报