DateFormat 类是一个非线程安全的类。javadocs 文档里面提到:"Date formats是不能同步的。 我们建议为每个线程创建独立的日期格式。 如果多个线程同时访问一个日期格式,这需要在外部加上同步代码块。"
如何并发使用DateFormat类?
1. 同步
最简单的方法就是在做日期转换之前,为DateFormat对象加锁。这种方法使得一次只能让一个线程访问DateFormat对象,而其他线程只能等待。
public class DateUtil { private static final String MESSAGE_FORMAT = "MM-dd HH:mm:ss.ms"; private static final SimpleDateFormat format=new SimpleDateFormat(MESSAGE_FORMAT, Locale.getDefault()); public static final DateFormat getDateFormat() { return format; } public Date parse(String str) { try { synchronized(format){ return format.parse(str); } } catch (ParseException e) { } return null; } }
这样没问题,但是加了锁,效率严重下降。
2.使用ThreadLocal的匿名内部类写法,推荐此方法
另外一个方法就是使用ThreadLocal变量去容纳DateFormat对象,也就是说每个线程都有一个属于自己的副本,并无需等待其他线程去释放它。这种方法会比使用同步块更高效。
public class DateUtil1 {
private static final ThreadLocal<DateFormat> df = new ThreadLocal<DateFormat>() {
@Override
protected DateFormat initialValue() {
return new SimpleDateFormat("yyyyMMdd");
}
};
public static Date convert(String source) throws ParseException {
Date d = df.get().parse(source);
return d;
}
}
阿里巴巴操作手册关于SimpleDateFormat的描述如下:
【强制】 SimpleDateFormat 是线程不安全的类,一般不要定义为 static 变量,如果定义为static ,必须加锁,或者使用 DateUtils 工具类。
正例:注意线程安全,使用 DateUtils 。亦推荐如下处理:
private static final ThreadLocal<DateFormat> df = new ThreadLocal<DateFormat>() { @ Override protected DateFormat initialValue() { return new SimpleDateFormat("yyyy-MM-dd"); } };
说明:如果是 JDK 8 的应用,可以使用 Instant 代替 Date , LocalDateTime 代替 Calendar ,DateTimeFormatter 代替 SimpleDateFormat ,官方给出的解释: simple beautiful strong immutable thread - safe 。
Java 8 已经普遍使用了,可是还在有人用 Java Calendar 处理时间和日期,不仅仅性能差,很切代码很冗余,就不能用 Java 8 提供的新 API 吗?所以 CTO 强制了,必须用 Java 8 处理日期,否则一律开除。下面是整理的 18 种处理日期的方式,可以收藏起来,一定有用。
Java处理日期、日历和时间的方式一直为社区所诟病,将 java.util.Date设定为可变类型,以及SimpleDateFormat的非线程安全使其应用非常受限。
新API基于ISO标准日历系统,java.time包下的所有类都是不可变类型而且线程安全。
示例1:Java 8中获取今天的日期
Java 8 中的 LocalDate 用于表示当天日期。和java.util.Date不同,它只有日期,不包含时间。当你仅需要表示日期时就用这个类。
import java.time.LocalDate; public class Demo01 { public static void main(String[] args) { LocalDate today = LocalDate.now(); System.out.println("今天的日期:"+today); } }
结果为:今天的日期:2022-08-24
示例2:Java 8中获取年、月、日信息
import java.time.LocalDate; public class Demo02 { public static void main(String[] args) { LocalDate today = LocalDate.now(); int year = today.getYear(); int month = today.getMonthValue(); int day = today.getDayOfMonth(); System.out.println("year:"+year); System.out.println("month:"+month); System.out.println("day:"+day); } }
结果如下:
year:2022 month:8 day:24
示例3:Java 8中处理特定日期
我们通过静态工厂方法now()非常容易地创建了当天日期,你还可以调用另一个有用的工厂方法LocalDate.of()创建任意日期, 该方法需要传入年、月、日做参数,返回对应的LocalDate实例。这个方法的好处是没再犯老API的设计错误,比如年度起始于1900,月份是从0开 始等等。
import java.time.LocalDate; public class Demo03 { public static void main(String[] args) { LocalDate date = LocalDate.of(2018,2,6); System.out.println("自定义日期:"+date); } }
结果:自定义日期:2018-02-06
示例4:Java 8中判断两个日期是否相等
import java.time.LocalDate; public class Demo04 { public static void main(String[] args) { LocalDate date1 = LocalDate.now(); LocalDate date2 = LocalDate.of(2018,2,5); if(date1.equals(date2)){ System.out.println("时间相等"); }else{ System.out.println("时间不等"); } } }
结果:时间不等
示例5:Java 8中检查像生日这种周期性事件
import java.time.LocalDate; import java.time.MonthDay; public class Demo05 { public static void main(String[] args) { LocalDate date1 = LocalDate.now(); LocalDate date2 = LocalDate.of(2018,2,6); MonthDay birthday = MonthDay.of(date2.getMonth(),date2.getDayOfMonth()); MonthDay currentMonthDay = MonthDay.from(date1); System.out.println(birthday); System.out.println(currentMonthDay); if(currentMonthDay.equals(birthday)){ System.out.println("是你的生日"); }else{ System.out.println("你的生日还没有到"); } } }
结果如下:
--02-06 --08-24 你的生日还没有到
只要当天的日期和生日匹配,无论是哪一年都会打印出祝贺信息。你可以把程序整合进系统时钟,看看生日时是否会受到提醒,或者写一个单元测试来检测代码是否运行正确。
示例6:Java 8中获取当前时间
import java.time.LocalTime; public class Demo06 { public static void main(String[] args) { LocalTime time = LocalTime.now(); System.out.println("获取当前的时间,不含有日期:"+time); } }
结果如下:获取当前的时间,不含有日期:11:24:20.506
可以看到当前时间就只包含时间信息,没有日期
示例7:Java 8中获取当前时间
通过增加小时、分、秒来计算将来的时间很常见。Java 8除了不变类型和线程安全的好处之外,还提供了更好的plusHours()方法替换add(),并且是兼容的。注意,这些方法返回一个全新的LocalTime实例,由于其不可变性,返回后一定要用变量赋值。
import java.time.LocalTime; public class Demo07 { public static void main(String[] args) { LocalTime time = LocalTime.now(); LocalTime newTime = time.plusHours(3); System.out.println("三个小时后的时间为:"+newTime); } }
结果:三个小时后的时间为:14:26:33.649
示例8:Java 8如何计算一周后的日期
和上个例子计算3小时以后的时间类似,这个例子会计算一周后的日期。LocalDate日期不包含时间信息,它的plus()方法用来增加天、周、月,ChronoUnit类声明了这些时间单位。由于LocalDate也是不变类型,返回后一定要用变量赋值。
import java.time.LocalDate; import java.time.temporal.ChronoUnit; public class Demo08 { public static void main(String[] args) { LocalDate today = LocalDate.now(); System.out.println("今天的日期为:"+today); LocalDate nextWeek = today.plus(1, ChronoUnit.WEEKS); System.out.println("一周后的日期为:"+nextWeek); } }
结果如下:
今天的日期为:2022-08-24 一周后的日期为:2022-08-31
可以看到新日期离当天日期是7天,也就是一周。你可以用同样的方法增加1个月、1年、1小时、1分钟甚至一个世纪,更多选项可以查看Java 8 API中的ChronoUnit类
示例9:Java 8计算一年前或一年后的日期
利用minus()方法计算一年前的日期
import java.time.LocalDate; import java.time.temporal.ChronoUnit; public class Demo09 { public static void main(String[] args) { LocalDate today = LocalDate.now(); LocalDate previousYear = today.minus(1, ChronoUnit.YEARS); System.out.println("一年前的日期 : " + previousYear); LocalDate nextYear = today.plus(1, ChronoUnit.YEARS); System.out.println("一年后的日期:"+nextYear); } }
结果如下:
一年前的日期 : 2021-08-24 一年后的日期:2023-08-24
示例10:Java 8的Clock时钟类
Java 8增加了一个Clock时钟类用于获取当时的时间戳,或当前时区下的日期时间信息。以前用到System.currentTimeInMillis()和TimeZone.getDefault()的地方都可用Clock替换。
import java.time.Clock; public class Demo10 { public static void main(String[] args) { // Returns the current time based on your system clock and set to UTC. Clock clock = Clock.systemUTC(); System.out.println("Clock : " + clock.millis()); // Returns time based on system clock zone Clock defaultClock = Clock.systemDefaultZone(); System.out.println("Clock : " + defaultClock.millis()); } }
结果如下:
Clock : 1661311871963 Clock : 1661311872026
示例11:如何用Java判断日期是早于还是晚于另一个日期
另一个工作中常见的操作就是如何判断给定的一个日期是大于某天还是小于某天?在Java 8中,LocalDate类有两类方法isBefore()和isAfter()用于比较日期。调用isBefore()方法时,如果给定日期小于当前日期则返回true。
import java.time.LocalDate; import java.time.temporal.ChronoUnit; public class Demo11 { public static void main(String[] args) { LocalDate today = LocalDate.now(); LocalDate tomorrow = LocalDate.of(2022,8,24); if(tomorrow.isAfter(today)){ System.out.println("之后的日期:"+tomorrow); } LocalDate yesterday = today.minus(1, ChronoUnit.DAYS); if(yesterday.isBefore(today)){ System.out.println("之前的日期:"+yesterday); } } }
结果如下:
之后的日期:2022-08-25 之前的日期:2022-08-23
示例12:Java 8中处理时区
Java 8不仅分离了日期和时间,也把时区分离出来了。现在有一系列单独的类如ZoneId来处理特定时区,ZoneDateTime类来表示某时区下的时间。这在Java 8以前都是 GregorianCalendar类来做的。下面这个例子展示了如何把本时区的时间转换成另一个时区的时间。
import java.time.LocalDateTime; import java.time.ZoneId; import java.time.ZonedDateTime; public class Demo12 { public static void main(String[] args) { // Date and time with timezone in Java 8 ZoneId america = ZoneId.of("America/New_York"); LocalDateTime localtDateAndTime = LocalDateTime.now(); System.out.println(localtDateAndTime); ZonedDateTime dateAndTimeInNewYork = ZonedDateTime.of(localtDateAndTime, america ); System.out.println("Current date and time in a particular timezone : " + dateAndTimeInNewYork); } }
结果如下:
2022-08-24T11:44:09.609 2022-08-24T11:44:09.609-04:00[America/New_York] Current date and time in a particular timezone : 2022-08-24T11:44:09.609-04:00[America/New_York]
示例13:如何表示信用卡到期这类固定日期,答案就在YearMonth
与 MonthDay检查重复事件的例子相似,YearMonth是另一个组合类,用于表示信用卡到期日、FD到期日、期货期权到期日等。还可以用这个类得到 当月共有多少天,YearMonth实例的lengthOfMonth()方法可以返回当月的天数,在判断2月有28天还是29天时非常有用。
import java.time.*; public class Demo13 { public static void main(String[] args) { YearMonth currentYearMonth = YearMonth.now(); System.out.printf("Days in month year %s: %d%n", currentYearMonth, currentYearMonth.lengthOfMonth()); YearMonth creditCardExpiry = YearMonth.of(2019, Month.FEBRUARY); System.out.printf("Your credit card expires on %s %n", creditCardExpiry); } }
结果如下:
Days in month year 2022-08: 31 Your credit card expires on 2019-02
示例14:如何在Java 8中检查闰年
import java.time.LocalDate; public class Demo14 { public static void main(String[] args) { LocalDate today = LocalDate.now(); if(today.isLeapYear()){ System.out.println("This year is Leap year"); }else { System.out.println("2018 is not a Leap year"); } } }
结果:2018 is not a Leap year
示例15:计算两个日期之间的天数和月数
有一个常见日期操作是计算两个日期之间的天数、周数或月数。在Java 8中可以用java.time.Period类来做计算。
下面这个例子中,我们计算了当天和将来某一天之间的月数。
import java.time.LocalDate; import java.time.Period; public class Demo15 { public static void main(String[] args) { LocalDate today = LocalDate.now(); LocalDate java8Release = LocalDate.of(2022, 12, 14); Period periodToNextJavaRelease = Period.between(today, java8Release); System.out.println("Months left between today and Java 8 release : " + periodToNextJavaRelease.getMonths() ); } }
结果:Months left between today and Java 8 release : 3
示例16:在Java 8中获取当前的时间戳
Instant类有一个静态工厂方法now()会返回当前的时间戳,如下所示:
import java.time.Instant; public class Demo16 { public static void main(String[] args) { Instant timestamp = Instant.now(); System.out.println("What is value of this instant " + timestamp.toEpochMilli()); } }
结果:What is value of this instant 1661313298160
时间戳信息里同时包含了日期和时间,这和java.util.Date很像。实际上Instant类确实等同于 Java 8之前的Date类,你可以使用Date类和Instant类各自的转换方法互相转换,例如:Date.from(Instant) 将Instant转换成java.util.Date,Date.toInstant()则是将Date类转换成Instant类。
示例17:Java 8中如何使用预定义的格式化工具去解析或格式化日期
import java.time.LocalDate; import java.time.format.DateTimeFormatter; public class Demo17 { public static void main(String[] args) { String dayAfterTommorrow = "20180205"; LocalDate formatted = LocalDate.parse(dayAfterTommorrow, DateTimeFormatter.BASIC_ISO_DATE); System.out.println(dayAfterTommorrow+" 格式化后的日期为: "+formatted); } }
结果:20220825 格式化后的日期为: 2022-08-25
示例18:字符串互转日期类型
import java.time.LocalDate; import java.time.LocalDateTime; import java.time.format.DateTimeFormatter; public class Demo18 { public static void main(String[] args) { LocalDateTime date = LocalDateTime.now(); DateTimeFormatter format1 = DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss"); //日期转字符串 String str = date.format(format1); System.out.println("日期转换为字符串:"+str); DateTimeFormatter format2 = DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss"); //字符串转日期 LocalDate date2 = LocalDate.parse(str,format2); System.out.println("日期类型:"+date2); } }
结果如下:
日期转换为字符串:2022/08/24 12:00:24 日期类型:2022-08-24
localDate转字符串
LocalDate date = LocalDate.now(); // 创建一个DateTimeFormatter实例来定义日期格式 DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd"); // 使用formatter转换LocalDate为字符串 String dateString = date.format(formatter);
示例19:将字符串类型的日期转成Date类型
public class demo { public static void main(String[] args) throws ParseException {// 日期字符串转LocalDateTime String today = "2022-08-24 18:00:10"; DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); // 转为自定义格式 LocalDateTime localDateTime = LocalDateTime.parse(today, dateTimeFormatter); String format = DateTimeFormatter.ofPattern("yyyyMMddHHmmss").format(localDateTime); System.out.println("localDateTime:" + localDateTime); System.out.println("自定义格式:" + format); } }
结果如下:
localDateTime:2022-08-24T18:00:10 自定义格式:20220824180010
示例20:获取今天的开始时间和结束时间
public class demo { public static void main(String[] args) throws ParseException { LocalDate today = LocalDate.now(); LocalDate yesterday = today.minus(1, ChronoUnit.DAYS); System.out.println(yesterday); LocalDateTime startOfDay = LocalDateTime.of(LocalDate.now(), LocalTime.MIN); LocalDateTime endOfDay = LocalDateTime.of(LocalDate.now(), LocalTime.MAX); System.out.println(startOfDay); System.out.println(endOfDay); LocalDateTime startOfDay1 = LocalDateTime.of(today.minus(1, ChronoUnit.DAYS), LocalTime.MIN); LocalDateTime endOfDay1 = LocalDateTime.of(today.minus(1, ChronoUnit.DAYS), LocalTime.MAX); System.out.println(startOfDay1); System.out.println(endOfDay1); } }
结果:
2022-08-28 2022-08-29T00:00 2022-08-29T23:59:59.999999999 2022-08-28T00:00 2022-08-28T23:59:59.999999999
常用转换
Calendar c = Calendar.getInstance(); //c.add(Calendar.DAY_OF_YEAR, -1); Date todayDate = c.getTime(); //Date yesterdayDate = c.getTime(); Instant instant = todayDate.toInstant(); // 使用 Instant 代替 Date ZoneId zoneId = ZoneId.systemDefault(); LocalDateTime localDateTime = instant.atZone(zoneId).toLocalDateTime(); // LocalDateTime 代替 Calendar System.out.println(localDateTime); // LocalDateTime 代替 Calendar ,DateTimeFormatter 代替 SimpleDateFormat DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd"); //String yesterdayDateStr = dateTimeFormatter.format(localDateTime); String todayDateStr = dateTimeFormatter.format(localDateTime); DateTimeFormatter dateTimeFormatter02 = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); String curTime = dateTimeFormatter02.format(localDateTime);
java.time.LocalDate --> java.util.Date
private Date LocalDateToDate(String date){ LocalDate localDate = LocalDate.parse(date); // LocalDate转换位Date ZoneId zoneId = ZoneId.systemDefault(); Instant instant = localDate.atStartOfDay().atZone(zoneId).toInstant(); return Date.from(instant); }
java.util.Date --> java.time.LocalDate
public void DateToLocalDate() { java.util.Date date = new java.util.Date(); Instant instant = date.toInstant(); ZoneId zone = ZoneId.systemDefault(); LocalDateTime localDateTime = LocalDateTime.ofInstant(instant, zone); LocalDate localDate = localDateTime.toLocalDate(); }
java.util.Date --> java.time.LocalDateTime
public void DateToLocalDateTime() { java.util.Date date = new java.util.Date(); Instant instant = date.toInstant(); ZoneId zone = ZoneId.systemDefault(); LocalDateTime localDateTime = LocalDateTime.ofInstant(instant, zone); }
java.time.LocalDateTime --> java.util.Date
public void LocalDateTimeTodate() { LocalDateTime localDateTime = LocalDateTime.now(); ZoneId zone = ZoneId.systemDefault(); Instant instant = localDateTime.atZone(zone).toInstant(); java.util.Date date = Date.from(instant); }
java.util.Date --> java.time.LocalTime
public void DateToLocalTime() { java.util.Date date = new java.util.Date(); Instant instant = date.toInstant(); ZoneId zone = ZoneId.systemDefault(); LocalDateTime localDateTime = LocalDateTime.ofInstant(instant, zone); LocalTime localTime = localDateTime.toLocalTime(); }
java.time.LocalTime --> java.util.Date
public void LocalTimeTodate() { LocalTime localTime = LocalTime.now(); LocalDate localDate = LocalDate.now(); LocalDateTime localDateTime = LocalDateTime.of(localDate, localTime); ZoneId zone = ZoneId.systemDefault(); Instant instant = localDateTime.atZone(zone).toInstant(); java.util.Date date = Date.from(instant); }
字符串(yyyy-MM-dd HH:mm:ss)转LocalDateTime
LocalDateTime localDate=LocalDateTime.parse(str, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
时间戳转日期
Date date = new Date(Long.parseLong("1657508223")); LocalDateTime.ofInstant(Instant.ofEpochMilli(date.getTime()), ZoneId.systemDefault()) .format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"))
日期转时间戳
//yyyy-MM-dd格式转时间戳 LocalDate localDate = LocalDate.parse("2022-01-01", DateTimeFormatter.ofPattern("yyyy-MM-dd")); LocalDateTime localDateTime = localDate.atStartOfDay(); Long time = localDateTime.toInstant(ZoneOffset.ofHours(8)).toEpochMilli(); //yyyy-MM-dd HH:mm:ss格式转时间戳 LocalDateTime localDate = LocalDateTime .parse("2022-01-01 10:09:09", DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")); Long time = localDate.toInstant(ZoneOffset.ofHours(8)).toEpochMilli();
3.Apache commons-lang中的FastDateFormat
private String initDate() { Date d = new Date(); FastDateFormat fdf = FastDateFormat.getInstance("yyyy-MM-dd HH:mm:ss"); return fdf.format(d); }