66

I have a Java class that takes in the latitude/longitude of a location and returns the GMT offset when daylight savings time is on and off. I am looking for an easy way to determine in Java if the current date is in daylight savings time so I can apply the correct offset. Currently I am only performing this calculation for U.S. timezones although eventually I would like to expand this to global timezones as well.

Lance Roberts
  • 21,279
  • 29
  • 106
  • 128
gelgamil
  • 841
  • 1
  • 7
  • 6
  • Currently I am retrieving the time zone information using the GeoTools library and a shapefile provided by the National Atlas of the United States (http://nationalatlas.gov/mld/timeznp.html). Fortunately this provides me with some additional information - primarily the time zone symbol which is 2 or 4 digits (i.e AL, EA, EAno, etc). Unfortunately this value doesn't correspond to those used by Java time zones although I could perform this mapping manually. Ideally I'd like a solution that would work if I replaced this file with a world time zone shapefile but that might be too ambitious. – gelgamil Jun 29 '09 at 21:21
  • timeznp020.txt lists `Enumerated_Domain`s which include details of each abbreviation used; it shouldn't be too difficult to map those to zoneinfo time zone names. I'm not sure if time zone names in Java are cross-platform, but I'd hope so! – tc. Oct 03 '11 at 23:52
  • another answer which is useful for this problem http://stackoverflow.com/a/1449510/311525 – Scott May 07 '14 at 05:18
  • http://stackoverflow.com/questions/10545960/how-to-tackle-daylight-savings-using-timezone-in-java –  Apr 08 '15 at 05:09

5 Answers5

76

This is the answer for the machine on which the question is being asked:

TimeZone.getDefault().inDaylightTime( new Date() );

A server trying to figure this out for a client will need the client's time zone. See @Powerlord answer for the reason why.

For any particular TimeZone

TimeZone.getTimeZone( "US/Alaska").inDaylightTime( new Date() );
Powerlord
  • 82,184
  • 16
  • 119
  • 164
Clint
  • 8,717
  • 1
  • 24
  • 32
49

tl;dr

ZoneId.of( "America/Montreal" )  // Represent a specific time zone, the history of past, present, and future changes to the offset-from-UTC used by the people of a certain region.  
      .getRules()                // Obtain the list of those changes in offset. 
      .isDaylightSavings(        // See if the people of this region are observing Daylight Saving Time at a specific moment.
          Instant.now()          // Specify the moment. Here we capture the current moment at runtime. 
      )                          // Returns a boolean.

java.time

Here is the modern java.time (see Tutorial) version of the correct Answer by mamboking.

Example code:

ZonedDateTime now = ZonedDateTime.now( ZoneId.of( "America/Montreal" ) );
…
ZoneId z = now.getZone();
ZoneRules zoneRules = z.getRules();
Boolean isDst = zoneRules.isDaylightSavings( now.toInstant() );

Note how in the last line we had to extract an Instant object from our ZonedDateTime object with a simple call to toInstant.

Table of date-time types in Java, both modern and legacy.


About java.time

The java.time framework is built into Java 8 and later. These classes supplant the troublesome old legacy date-time classes such as java.util.Date, Calendar, & SimpleDateFormat.

The Joda-Time project, now in maintenance mode, advises migration to the java.time classes.

To learn more, see the Oracle Tutorial. And search Stack Overflow for many examples and explanations. Specification is JSR 310.

You may exchange java.time objects directly with your database. Use a JDBC driver compliant with JDBC 4.2 or later. No need for strings, no need for java.sql.* classes.

Where to obtain the java.time classes?

The ThreeTen-Extra project extends java.time with additional classes. This project is a proving ground for possible future additions to java.time. You may find some useful classes here such as Interval, YearWeek, YearQuarter, and more.

Basil Bourque
  • 218,480
  • 72
  • 657
  • 915
10
TimeZone tz = TimeZone.getTimeZone("EST");
boolean inDs = tz.inDaylightTime(new Date());
mamboking
  • 4,239
  • 20
  • 27
  • 7
    Three later abbreviations like EST (meant for standard times) should be avoided as they are DST-insensible. Use a time zone id instead. – Tiny Jan 15 '16 at 19:51
3

You're going to have to do a bit more work using those coordinates and figure out which time zone they're in. Once you know which TimeZone that is, the isDayLight() method would be useful.

For example, you have no way of telling whether -0500 is EST (US/Canada Eastern Standard Time), CDT (US/Canada Central Daylight Time), COT (Colombia Time), AST (Brazil Acre Standard Time), ECT (Ecuador Time), etc...

Some of these may or may not support daylight saving time.

AngocA
  • 7,352
  • 4
  • 34
  • 51
Powerlord
  • 82,184
  • 16
  • 119
  • 164
1

Joda Time contains handling methods which will calculate the offsets for you. See DateTimeZone.convertLocalToUTC(...)

To supplement this, you will need to look up the current time zone with your latitude/longitude info. GeoNames provides a java client for its web service, as well as a simple web-request framework (i.e. http://ws.geonames.org/timezone?lat=47.01&lng=10.2)

James Van Huis
  • 5,333
  • 1
  • 24
  • 25
  • I've looked into the GeoNames web service previously and I am aware it will accomplish what I am trying to do. Unfortunately this code will be hosted on servers that may be located behind a firewall so I'm trying to avoid external web service calls if possible. For those without this restriction the GeoNames web service would be a great solution. – gelgamil Jun 29 '09 at 21:34