In addition to lambda expressions, Streams, and several minor improvements, Java 8 introduces a new set of date and time apis. In this tutorial, we’ll walk through a few simple task examples to learn how to use the Java 8 API. Java has been criticized for its handling of dates, calendars, and times, particularly its decision to make java.util.Date modifiable and SimpleDateFormat non-thread-safe.
It seems that Java has realized the need to provide better support for time and date functions, which is also a good thing for the community that has grown accustomed to using the Joda time and date library. The best thing about this new date and time library is that it defines concepts related to time and date, such as Instant, duration, date, time, time-zone, and Period. It also borrows some of the strengths of the Joda library, such as its ability to separate human and machine understanding of time and date. Java 8 still uses the ISO calendar architecture and, unlike its predecessors, the classes in the Java.time package are immutable and thread-safe. The new time and date API is in the java.time package. Here are some of the key classes in it:
- Instant — This stands for timestamp
- LocalDate – a date that does not contain a specific time, such as 2014-01-14. It can be used to store birthdays, anniversaries, entry dates, etc.
- LocalTime – this stands for dateless time
- LocalDateTime – This contains the date and time, but still no offset information or time zone.
- ZonedDateTime – This is a complete datetime with a time zone, offset in UTC/ Greenwich Mean Time.
The new library also adds ZoneOffset and Zoned, which provide better support for time zones. With the new DateTimeFormatter, date formatting has taken on a new look. By the way, I wrote this article when Java was launching this new feature this time last year, so you’ll notice that the dates in the examples are all from last year. If you run these examples, they will definitely return the correct value.
How does Java 8 handle time and dates
I was asked what is the best way to learn a new library? My answer is, just use it that way in a real project. In a real project, there would be a variety of requirements that would prompt developers to explore and explore this new library. In short, it’s the task itself that really motivates you to explore and learn. The same goes for Java 8’s new date and time API. To learn about this new Java 8 library, I’ve created 20 task-oriented examples here. We begin with a simple task, such as how to use Java 8 time date library to show today, and then further to generate a timely area with complete date, then study how to complete some of the more practical task, such as development of a reminder application, to find out from some specific date such as birthday, anniversary, on Sunday The number of days until the next billing date, premium date or credit card expiration date.
Example 1: How do I get the date of the day in Java 8
Java 8 has a class called LocalDate that can be used to represent today’s date. This class is slightly different from java.util.date in that it contains only dates and no time. So, if you only need to represent the date and not the time, you can use it.
LocalDate today = LocalDate.now();
System.out.println("Today's Local date : "+ today); The output Today's Local date : 2018-02-11
Copy the code
You can see that it creates today’s date but contains no time information. It also formats the Date before printing it out, unlike the previous Date class, which printed the data unformatted.
Example 2: How do I get the current year, month and day in Java 8
The LocalDate class provides some handy methods for extracting the year, month, day and other date attributes. Using these methods, you can get any date attributes you want without using classes like java.util.calendar:
LocalDate today = LocalDate.now();
int year = today.getYear();
int month = today.getMonthValue();
int day = today.getDayOfMonth();
System.out.printf("Year : %d Month : %d day : %d \t %n", year, month, day); The output Today's Local date : 2018-02-11 Year : 2018 Month : 2 day : 11Copy the code
As you can see, getting the date information in Java 8 is very simple, just use the corresponding getter method, no memory, very straightforward. You can compare this to the old Java way of getting the current year, month and day.
Example 3: How do I get a particular date in Java 8
In the first example, we saw how simple it is to generate the date of the day with the static method now(), but with another useful factory method, localdate.of (), we can create an arbitrary date that takes a year, month and day and returns an equivalent LocalDate instance. The other good news about this method is that it does not repeat the mistakes of the previous API, for example, the year can only start from 1900, the month must start from 0, etc. The date here is whatever you write, for example, in this case it’s January 14th, there’s no hidden logic.
LocalDate dateOfBirth = LocalDate.of(2010, 01, 14);
System.out.println("Your Date of birth is : "+ dateOfBirth); Your Date of birth is: 2010-01-14Copy the code
As you can see, the created date is as we wrote it, January 14, 2014.
Example 4 how do I check if two dates are equal in Java 8
When it comes to real time and date tasks, a common one is to check whether two dates are equal. You may often find yourself trying to decide if today is a special occasion, such as a birthday, anniversary, or holiday. Sometimes you’ll be given a date and asked to check if it’s a date like a holiday. The following example will help you do this in Java 8. As you might expect, LocalDate overrides the Equals method to compare dates, as shown below:
LocalDate date1 = LocalDate.of(2014, 01, 14);
if(date1.equals(today)){
System.out.printf("Today %s and date1 %s are same date %n", today, date1); } today 2014-01-14 and date1 2014-01-14 are same dateCopy the code
In this case, the two dates we are comparing are equal. Also, if you get a formatted date string in code, you have to parse it into dates before you can compare. You can compare this example to the way Java used to compare dates, and you’ll see how much better it is.
Example 5. How do I check for repeated events, such as birthdays, in Java 8
Another practical time and date-related task in Java is to check for recurring events, such as monthly billing dates, wedding anniversaries, monthly payment dates, or annual insurance payment dates. If you work in an e-commerce company, there will be a module that sends birthday greetings to users and sends greetings to them on every important holiday, such as Christmas, Thanksgiving, or Deepawali in India. How do I determine if it is a holiday or repeated event in Java? Use the MonthDay class. This class consists of a combination of month and day and contains no year information, meaning you can use it to represent days that repeat each year. There are other combinations, such as the YearMonth class. It is immutable and thread-safe like the other classes in the new date and time library, and it is a value class. Let’s look at an example of how to use MonthDay to check a duplicate date:
LocalDate dateOfBirth = LocalDate.of(2010, 01, 14);
MonthDay birthday = MonthDay.of(dateOfBirth.getMonth(), dateOfBirth.getDayOfMonth());
MonthDay currentMonthDay = MonthDay.from(today);
if(currentMonthDay.equals(birthday)){
System.out.println("Many Many happy returns of the day !!");
}else{
System.out.println("Sorry, today is not your birthday"); } Output: Many Many happy returns of the day!!Copy the code
Although the year is different, today is your birthday, so you will see a birthday greeting in the output. You can adjust the system timing and run the program to see if it reminds you when your next birthday is. You can also try writing a JUnit unit test on your next birthday to see if the code works.
Example 6 how do I get the current time in Java 8
This is very similar to getting the current date in the first example. This time we’re using a class called LocalTime, which is a dateless time and a close relative of LocalDate. Here you can also use the static factory method now() to get the current time. The default format is HH :mm:ss: NNN, where NNN is nanosecond. Compare this to how Java 8 used to get the current time.
LocalTime time = LocalTime.now();
System.out.println("local time now : "+ time); The outputlocalTime now: 16:33:33.369 //in hour, minutes, seconds, nano seconds
Copy the code
As you can see, the current time does not contain a date, because LocalTime only has a time and no date.
Example 7 How to increase the number of hours in the time
Many times we need to add hours, minutes, or seconds to calculate future time. Not only does Java 8 offer immutable, thread-safe classes, it also offers more convenient methods such as plusHours() to replace the old add() method. By the way, these methods return a reference to a new LocalTime instance, because LocalTime is immutable, so don’t forget to store the new reference.
LocalTime time = LocalTime.now();
LocalTime newTime = time.plusHours(2); // adding two hours
System.out.println("Time after 2 hours : "+ newTime); Output: Time after 2 hours: 18:33:33.369Copy the code
You can see that the current time is 16:33:33.369 in 2 hours. Now you can compare it to the old way of adding or subtracting hours in Java. You can see which way is better.
Example 8. How do I obtain the date one week later
This is similar to the previous example of getting the date after 2 hours. Here we will learn how to get the date after 1 week. LocalDate is used to represent dates without time. It has a plus() method for adding days, weeks, or months. ChronoUnit is used to represent units of time. Since LocalDate is also immutable, any modification will return a new instance, so don’t forget to save it.
LocalDate nextWeek = today.plus(1, ChronoUnit.WEEKS);
System.out.println("Today is : " + today);
System.out.println("Date after 1 week : "+ nextWeek); Today is: 2018-01-14 Date after 1 week: 2018-01-21Copy the code
You can see what the date is seven days from now, a week from now. You can use this method to add a month, a year, an hour, a minute, or even a decade. Check out the ChronoUnit class in the Java API for more options.
Example 9 Date before or after one year
This is a sequel to the last example. In the previous example, we learned how to use LocalDate plus() to add a day, week, or month to a date. Now we’ll learn how to use minus() to find the date a year ago.
LocalDate previousYear = today.minus(1, ChronoUnit.YEARS);
System.out.println("Date before 1 year : " + previousYear);
LocalDate nextYear = today.plus(1, YEARS);
System.out.println("Date after 1 year : "+ nextYear); Date before 1 year: 2013-01-14 Date after 1 year: 2015-01-14Copy the code
It can be seen that there are two years in total, one is 2013 and the other is 2015, respectively before and after 2014.
Example 10 uses clocks in Java 8
Java 8 comes with a Clock class that you can use to get the current instantaneous time, date, or time in a particular time zone. Can use a Clock to replace System. CurrentTimeInMillis () and TimeZone. GetDefault () method.
// Returns the current time based on your system clock and set to UTC.
Clock clock = Clock.systemUTC();
System.out.println("Clock : " + clock);
// Returns time based on system clock zone Clock defaultClock =
Clock.systemDefaultZone();
System.out.println("Clock : "+ clock); Clock: SystemClock[Z] Clock: SystemClock[Z]Copy the code
You can compare the clock to the specified date, as follows:
public class MyClass {
private Clock clock; // dependency inject ...
public void process(LocalDate eventDate) {
if(eventDate.isBefore(LocalDate.now(clock)) { ... }}}Copy the code
This is handy if you need to deal with dates in different time zones.
Example 11 How do I determine in Java whether a date comes before or after another date
This is also a common task in real life projects. How do you tell if a date is before or after another date, or even exactly the same? In Java 8, the LocalDate class has an isBefore() and isAfter() method that can be used to compare two dates. The isBefore() method returns true if the date on which the method was called is earlier than the given date.
LocalDate tomorrow = LocalDate.of(2014, 1, 15); ,if(tommorow.isAfter(today)){
System.out.println("Tomorrow comes after today");
}
LocalDate yesterday = today.minus(1, DAYS);
if(yesterday.isBefore(today)){
System.out.println("Yesterday is day before today"); } Tomorrow comes after today Yesterday is day before todayCopy the code
You can see how easy it is to do date comparisons in Java 8. You don’t need another class like Calendar to do similar tasks.
Example 12 deals with different time zones in Java 8
Java 8 not only separates date and time, but also time zones. There are already several classes related to time zones, such as ZonId for a specific time zone and ZonedDateTime for time with a time zone. It is equivalent to the GregorianCalendar class prior to Java 8. Using this class, you can convert the local time to the corresponding time in another time zone, as in this example:
// Date and time with timezone in Java 8 ZoneId america = ZoneId.of("America/New_York");
LocalDateTime localtDateAndTime = LocalDateTime.now();
ZonedDateTime dateAndTimeInNewYork = ZonedDateTime.of(localtDateAndTime, america );
System.out.println("Current date and time in a particular timezone : "+ dateAndTimeInNewYork); Output: Current date and timeinA particular timezone: 2014-01-14T16:33:33.373-05:00[America/New_York]Copy the code
Compare this to the previous method of converting local time to GMT time. By the way, as before Java 8, don’t be mistaken about the text corresponding to the time zone, or you’ll get an exception like this:
Exception in thread "main" java.time.zone.ZoneRulesException: Unknown time-zone ID: ASIA/Tokyo
at java.time.zone.ZoneRulesProvider.getProvider(ZoneRulesProvider.java:272)
at java.time.zone.ZoneRulesProvider.getRules(ZoneRulesProvider.java:227)
at java.time.ZoneRegion.ofId(ZoneRegion.java:120)
at java.time.ZoneId.of(ZoneId.java:403)
at java.time.ZoneId.of(ZoneId.java:351)
Copy the code
How does example 13 represent a fixed date, such as when a credit card expires
Just as MonthDay is a MonthDay that is a recurring date, YearMonth is another combination of dates like credit card payment date, time deposit due date, and Options due date. You can use this class to find out how many days the month has. The lengthOfMonth() method returns the number of days the YearMonth instance has, which is useful for checking whether February is 28 or 29 days.
YearMonth currentYearMonth = YearMonth.now(); System.out.printf("Days in month year %s: %d%n", currentYearMonth, currentYearMonth.lengthOfMonth());
YearMonth creditCardExpiry = YearMonth.of(2018, Month.FEBRUARY);
System.out.printf("Your credit card expires on %s %n", creditCardExpiry); Output: Daysin month year 2014-01: 31
Your credit card expires on 2018-02
Copy the code
Example 14. How do I check leap years in Java 8
The LocalDate class has an isLeapYear() method that returns whether the current LocalDate corresponds to a leap year. If you want to repeat the wheel, you can look at this code, which is pure Java logic for determining whether a year is a leap year.
if(today.isLeapYear()){
System.out.println("This year is Leap year");
}else {
System.out.println("2018 is not a Leap year"); } Output: 2018 is not a Leap yearCopy the code
You can check the results over several years to see if they are correct, and it is best to write a unit test to test normal years and leap years.
Example 15: How many days and how many months are included between two dates
Another common task is to calculate how many days, weeks, or years lie between two given dates. You can do this with the java.time.Period class. In the following example, we count the number of months between the current date and a future date.
LocalDate java8Release = LocalDate.of(2014, Month.MARCH, 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: 2Copy the code
As you can see, this month is January, and the Java 8 release date is March, so there is a 2-month gap.
Example 16 Date and time with time zone offset
In Java 8, you can use the ZoneOffset class to represent a time zone, such as GMT or UTC5:30 for India, and you can use its static zoneoffset.of () method to get the corresponding time zone. Once you get the offset, you can create an OffsetDateTime with LocalDateTime and the offset.
LocalDateTime datetime = LocalDateTime.of(2014, Month.JANUARY, 14, 19, 30);
ZoneOffset offset = ZoneOffset.of("+ 05:30");
OffsetDateTime date = OffsetDateTime.of(datetime, offset);
System.out.println("Date and Time with timezone offset in Java : "+ date); Date and Time with timezone offsetin Java : 2014-01-14T19:30+05:30
Copy the code
You can see that the time and date are now associated with the time zone. Another point is that OffSetDateTime is primarily intended for machines to understand. If it is intended for humans, use the ZoneDateTime class.
Example 17 How do I get the current timestamp in Java 8
If you remember how to get the current timestamp before Java 8, it’s a piece of cake now. The Instant class has a static factory method now() that returns the current timestamp as follows:
Instant timestamp = Instant.now();
System.out.println("What is value of this instant "+ timestamp); What is value of this instant 2014-01-14T08:33:33.379zCopy the code
Instant is a pre-Java 8 Date. You can use methods in both classes to convert between the two types. For example, date.from (Instant) is used to convert Instant to java.util.date, while date.toinstant () converts Date toInstant.
Example 18 how do I parse/format dates in Java 8 using a predefined formatter
Before Java 8, formatting dates and times was a technical matter, and our good friend SimpleDateFormat wasn’t thread-safe and was a bit cumbersome to format as a local variable. Thanks to thread-local variables, this makes it useful in multithreaded environments, but Java has maintained this state for a long time. This time it introduces a new thread-safe date and time formatter. It also comes with some pre-defined formatters that include commonly used date formats. For example, in this example we use the pre-defined BASIC_ISO_DATE format, which formats February 14, 2014 to 20140114.
String dayAfterTommorrow = "20140116";
LocalDate formatted = LocalDate.parse(dayAfterTommorrow,
DateTimeFormatter.BASIC_ISO_DATE);
System.out.printf("Date generated from String %s is %s %n", dayAfterTommorrow, formatted); Date generated from String 20140116 is 2014-01-16Copy the code
You can see that the generated date matches the value of the specified string, but the date format is slightly different.
Example 19. How do I use a custom formatter to parse dates in Java
In the example above, we used a built-in date and time formatter to parse the date string. Of course, predefined date formatters are great, but sometimes you still need to use custom date formatters. In this case, you need to create a custom date formatter instance yourself. In this example, the date format is “MMM dd YYYY”. You can pass any pattern to the ofPattern static method () of the DateTimeFormatter, which returns an instance with the same literal as in the previous example. For example, M is still the month, and M is still the minute. An invalid schema will throw a DateTimeParseException, but if it is a logical error such as using M when M should be used, there is nothing to be done.
String goodFriday = "Apr 18 2014";
try {
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("MMM dd yyyy");
LocalDate holiday = LocalDate.parse(goodFriday, formatter);
System.out.printf("Successfully parsed String %s, date is %s%n", goodFriday, holiday);
} catch (DateTimeParseException ex) {
System.out.printf("%s is not parsable! %n", goodFriday); ex.printStackTrace(); } Successfully parsed String Apr 18 2014, date is 2014-04-18Copy the code
You can see that the date value does match the string passed in, but in a different format.
Example 20: How do I format dates into strings in Java 8
In the last two examples, although we used the DateTimeFormatter class, we mainly parsed the date string. In this case, we’re going to do the opposite. Here we have an instance of the LocalDateTime class, which we will convert to a formatted date string. This is by far the easiest way to convert dates to strings in Java. The following example will return a formatted string. As in the previous example, we still need to create an instance of the DateTimeFormatter class using the specified pattern string, but instead of calling the LocalDate class’s parse method, we call its Format () method. This method returns a string representing the current date with the pattern defined in the DateTimeFormatter instance passed in.
LocalDateTime arrivalDate = LocalDateTime.now();
try {
DateTimeFormatter format = DateTimeFormatter.ofPattern("MMM dd yyyy hh:mm a");
String landing = arrivalDate.format(format);
System.out.printf("Arriving at : %s %n", landing);
} catch (DateTimeException ex) {
System.out.printf("%s can't be formatted! %n", arrivalDate);
ex.printStackTrace();
}
Output : Arriving at : Jan 14 2014 04:33 PM
Copy the code
As you can see, the current time is represented by the given “MMM dd YYYY HH :mm A” pattern, which contains the three letter months and the time represented by AM and PM.
A few key points of the Date and Time API in Java 8
After reading these examples, I’m sure you have some idea of Java 8’s new date and time apis. Now let’s review some of the key elements of this new API.
- It provides javax.time.zoneid for handling time zones.
- It provides LocalDate and LocalTime classes
- All classes in the new Date and time API in Java 8 are immutable and thread-safe, as opposed to the previous Date and Calendar API, where key classes like Java.util.Date and SimpleDateFormat are not thread-safe.
- An important aspect of the new Time and Date API is that it defines basic time and date concepts, such as instantaneous time, duration, date, time, time zone, and time period. They are all based on the ISO calendar system.
- Every Java developer should know at least five of these classes in the new API:
- Instant represents a timestamp, such as 2014-01-14T2:20:13.592z, which can be retrieved from the java.time.Clock class, like this: Instant current = Clock. System (ZoneId) of (” Asia/Tokyo “). The Instant ();
- LocalDate indicates a date with no time, such as 2014-01-14. It can be used to store birthdays, anniversaries, entry dates, etc.
- LocalTime – This represents the time without the date
- LocalDateTime – This contains the time and date, but no offset with time zone
- ZonedDateTime – This is the full time with a time zone, adjusted according to UTC/ Greenwich Mean Time
- The main package of this library is java.time, which contains classes for date, time, instant, and duration. It has two subpackages, java.time.foramt, which is obvious for what purpose, and java.time.temporal, which accesses fields at a lower level.
- Time zones refer to the regions of the earth that share the same standard time. Each time zone has a unique identifier, along with a region/city (Asia/Tokyo) format and an offset time from Greenwich Mean Time. For example, the offset time for Tokyo is +09:00.
- The OffsetDateTime class actually contains LocalDateTime and ZoneOffset. It is used to represent a complete date (year, month, day) and time (hour, minute, nanosecond) with an offset from Greenwich Mean Time (+/- hour: minute, such as +06:00 or -08:00).
- The DateTimeFormatter class is used to format and parse dates in Java. Unlike SimpleDateFormat, it is immutable and thread-safe and can be assigned to a static variable if needed. The DateTimeFormatter class provides a number of predefined formatters, and you can customize your own format as well. Of course, by convention, it also has a parse() method for converting a string to a date, which throws a DateTimeParseException if anything goes wrong during the conversion. Similarly, the DateFormatter class has a format() method for formatting dates, which throws a DateTimeException if it fails.
- “MMM d YYYY” and “MMM DD YYYY” are also slightly different. The former recognizes “Jan 2 2014” and “Jan 14 2014”, while the latter will report an error if “Jan 2 2014” is passed in. Because it expects two characters at the month. To solve this problem, if the day is one digit, you have to fill in the 0 in front of it. For example, “Jan 2 2014” should be changed to “Jan 02 2014”.
So much for Java 8’s new date and time API. These brief examples are sufficient to understand some of the new classes in the new API. Since it’s based on actual tasks, you don’t have to look around for time and date processing in Java. We learned how to create and modify date instances. We also learned about the difference between pure date, date plus time, date plus time, how to compare two dates, how to find out how many days there are between a certain date and a certain date like your next birthday, anniversary or insurance date. We also learned how to parse and format dates in Java 8 in a thread-safe way, without using thread-local variables or third-party libraries to cheat. The new API can handle any task related to time and date.
原 文 :
A flower has a first
Original source:
javarevisited