I need to extract the date field from DB and store it in a VO. How can I compare the hours difference from two dates.
For ex:
Let's say date1 = 01-SEP-17 10:00:00
and date2 = 05-SEP-17 12:00:00
. I need to compare the two dates and perform some operations like:
if(hours>10){
//do something
}
if(hours<10){
//do something else
}
I'm just able to calculate the difference between the hours (date2-date1) as 2 but how to consider the date too while calculating the difference between the hours?
My present code:
Date dateA = someVO.getDate();
long date = System.currentTimeMillis();
SimpleDateFormat df = new SimpleDateFormat("dd-MM-YY HH:mm:ss");
Date date1 = new Date(date);
Date date2 = df.parse(dateA.toString());
long date1Hours = date1.getHours();
long date2Hours = date2.getHours();
long dateDiff = date1Hours-date2Hours;
if(dateDiff>10){
//something
}
else if(dateDiff<10){
//something else
}
First you need to change the pattern used in
SimpleDateFormat
, and also use ajava.util.Locale
to specify that the month name is in English (otherwise it uses the system default locale, and it's not guaranteed to always be English).Then you get the correspondent millis value of each
Date
, calculate the difference between them and convert this to hours, using ajava.util.concurrent.TimeUnit
:dateDiff
will be98
.If you want to compare with the current date, just use
new Date()
.Daylight Saving Time issues
There's one problem with this approach. Although it doesn't make a difference for most part of the year, there can be differences due to Daylight Saving Time changes.
By default,
SimpleDateFormat
uses the JVM default timezone. If between the 2 dates there's a Daylight Saving Time changeover (or just an offset change), the result might be different.Example: in
Africa/Windhoek
timezone, in September 3rd 2017, at 2 AM, clocks shifted 1 hour forward, from 2 AM to 3 AM (and the offset changed from+01:00
to+02:00
). This means that, at that day, all local times between 2 AM and 2:59 AM don't exist in this timezone (it's like they "skipped" this hour).So, if the JVM default timezone is
Africa/Windhoek
, then the difference using the code above will be 97 hours (and not 98).Even if your JVM default timezone is not
Africa/Windhoek
, this can still happen, depending on the timezone and the dates involved. Not only that, but the default timezone can be changed without notice, even at runtime. It's always better to specify which timezone you're working with instead of just relying on the default.You can't avoid DST effects (unless you use UTC), but at least you can choose which timezone you're going to use instead of relying on the system default (that can be changed without notice).
It's possible to set a timezone in the formatter, so all dates will be parsed taking this timezone into account. In the example below, I'm using
Europe/London
, but of course you can change to one that best suits your case:Now all the parsed dates will be considered to be in London timezone (but remind that DST effects will still be considered - the advantage is that you know what timezone you're using and any changes in the JVM's default won't make your code suddenly start giving different and unexpected results).
Always use IANA timezones names (always in the format
Continent/City
, likeAmerica/Sao_Paulo
orEurope/Berlin
). Avoid using the 3-letter abbreviations (likeCST
orPST
) because they are ambiguous and not standard.You can get a list of all timezones using
TimeZone.getAvailableIDs()
- then you can choose the one that best suits your case.If you don't want to consider DST effects, you can use
TimeZone.getTimeZone("UTC")
- because UTC is a standard without DST changes.Java new Date/Time API
The old classes (
Date
,Calendar
andSimpleDateFormat
) have lots of problems and design issues, and they're being replaced by the new APIs.If you're using Java 8, consider using the new java.time API. It's easier, less bugged and less error-prone than the old APIs.
If you're using Java <= 7, you can use the ThreeTen Backport, a great backport for Java 8's new date/time classes. And for Android, there's the ThreeTenABP (more on how to use it here).
The code below works for both. The only difference is the package names (in Java 8 is
java.time
and in ThreeTen Backport (or Android's ThreeTenABP) isorg.threeten.bp
), but the classes and methods names are the same.First you need to parse the inputs (using a
DateTimeFormatter
) and specify in what timezone they are. As the dates also have a timezone, I'm using aZonedDateTime
, which is the best choice for this case.Then you can easily calculate the difference in hours using a
ChronoUnit
. In the example below, I'm also using London timezone as an example:If you want to use UTC, just change the
ZoneId
toZoneOffset.UTC
constant. If you want to compare with the current date, just use:Conversions to/from Date
If you still need to work with
java.util.Date
, it's possible to convert from/to the new API. In Java 8 you can use native methods, and in Java <=7 the ThreeTen Backport has theorg.threeten.bp.DateTimeUtils
class.To convert a
Date
to the new classes:To convert a
ZonedDateTime
back to a date: