How To Handle Dates In Java - metaruslan's blog
Home

How to Handle Dates in Java

Summary

Most tools/frameworks/databases give programmers poor choices for date types and offer very misleading behaviors. Even experienced programmers get confused and choose wrong types, what's worse - they take the misleading behaviors for granted. The biggest evil is unnecessary timezone conversions.

Timestamps and local dates - different paradigms

Timestamp

A timestamp is a moment on the time-line. It is always in UTC, because it doesn't need to be convenient for the end user, and because it needs to be unambiguous. In the Java world a timestamp can be represented as (ignoring milliseconds/nanoseconds precisions for simplicity):

Important! If you deal with a number of seconds/milliseconds/etc. then it must be in UTC, otherwise it would be too confusing. For example, java.lang.System.currentTimeMillis() in Java and date +%s command in Unix return milliseconds/seconds in UTC. Please drop me a comment if you know a tool or a framework that would deal with non-UTC numbers.

The representations above can be thought of as timezone-agnostic. You may disagree, because they do have a timezone, the UTC timezone, but UTC is a part of the definition, not a part of the data. The practical takeaway is that all of these invocations produce the same results on computers with different timezones:

System.currentTimeMillis()
new Date()
new Timestamp(System.currentTimeMillis())
Instant.now()
Instant.now().toString()

and we can convert between the types without specifying a timezone (or even without implicitly taking computer's timezone).

Local Date

A local date is a date/time from a human perspective, e.g. as seen on a wall clock. It's not good for computers, because it's impossible to convert it to an unambiguous moment on the time-line without knowing the timezone. Local dates can be represented as:

So, "1970-01-01T00:00:00Z" is a timestamp, "1970-01-01T00:00:00" is a local date, very different notions.

Unfortunately, tools/frameworks often confuse timestamps and local dates

They try to be user-friendly, but it almost always turns out awful. For example, consider the ill-behaved java.sql.Timestamp.toString() method. new java.sql.Timestamp(0L).toString() returns "1970-01-01 03:00:00.0" for the Moscow timezone and here is what actually happens: The consequences are:

In contrast, the modern equivalent java.time.Instant.ofEpochMilli(0).toString() would return "1970-01-01T00:00:00Z" irrespective of computer's timezone. Even if we want to switch to the local date paradigm - it will force us to provide an explicit timezone, e.g. Instant.ofEpochMilli(0).atZone(zoneId).toLocalDateTime() where zoneId can be, for example, ZoneId.systemDefault() or ZoneOffset.UTC.

It's not only the legacy Java classes problem

Consider MySQL's TIMESTAMP behavior: MySQL converts TIMESTAMP values from the current time zone to UTC for storage, and back from UTC to the current time zone for retrieval.

Avoid unnecessary timezone conversions

Guidelines

Concerns with java8 dates

This is what I don't understand or don't like personally, I'm sure there were reasons for the design choices, because java8 dates are really good.

Useful links

My popular stackoverflow answer regarding dates
Stackoverflow profile to see interesting answers from the author of Joda-Time who also worked on java8 times.

Comments