jueves, noviembre 13, 2008

Transformando fechas a diferentes zonas horarias (TimeZone)

Ya es sabido por todo programador Java, que uno de los puntos mas bajos, recae en el uso de las fechas, las mismas se encuentran super mal diseñadas y algunos objetos como el caso de Date, practicamente no son usables, pues toda su API esta deprecada (cosa que siento debería de dejar de ponerla deprecada, pues van por la versión 6 y aun la conservan). Recientemente me encontré con el siguiente problema; resulta que al poner un sistema en otro servidor, el cual aparentemente tiene una diferencia horaria configurada, obtenemos como seria de esperar resultados no esperados, cuando realizamos consultas con fechas a la base de datos. La primera solución que se nos ha ocurrido es implementar un convertidor de fechas a diferentes zonas horarias, a continuación coloco el método necesario para realizar la operación:


public static Date convertToTimeZoneDate(Date date, TimeZone timeZone) {

Date newTimeZoneDate = null;
Calendar foreignCalendar = null;
// Create a Calendar object with the local time zone
Calendar localCalendar = new GregorianCalendar();
localCalendar.setTime(date);

// Create an instance using foreign time zone and set it with the local
// UTC
strangerCalendar = new GregorianCalendar(timeZone);
strangerCalendar.set(Calendar.HOUR_OF_DAY, localCalendar
.get(Calendar.HOUR_OF_DAY));
strangerCalendar.set(Calendar.MINUTE, localCalendar
.get(Calendar.MINUTE));
strangerCalendar.set(Calendar.SECOND, localCalendar
.get(Calendar.SECOND));

// Create a Calendar object with the local time zone and set
// the UTC from foreign calendar
newTimeZoneDate = new Date(strangerCalendar.getTimeInMillis());

return newTimeZoneDate;
} // convertToTimeZone.

Este conveniente método, puede usarse de la siguiente forma:


String newTimeZoneId = "Chile/Continental";
Calendar calendar = new GregorianCalendar ();
Date date2 = calendar.getTime();
System.out.println(date2 + ":" + calendar.getTimeZone().getID());
System.out.println(convertToTimeZoneDate(date2, TimeZone
.getTimeZone(newTimeZoneId)) + ":" + newTimeZoneId);


Este código nos permite, obtener convertir nuestra fecha actual a la hora en chile. Para obtener la lista de las zonas horarias disponibles, puede ejecutar el siguiente código:


import java.io.PrintStream;
import java.util.Date;
import java.util.TimeZone;

public class TimeZoneAvailable {

/**
*
*/
public TimeZoneAvailable() {
}

public void execute() {

this.printTimeZones(System.out);
}

public void printTimeZones (PrintStream out) {

Date today = new Date();

// Get all time zone ids
String[] zoneIds = TimeZone.getAvailableIDs();

out.println("zoneIds.length = " + zoneIds.length);
// View every time zone
for (int i = 0; i < zoneIds.length; i++) {
// Get time zone by time zone id
TimeZone tz = TimeZone.getTimeZone(zoneIds[i]);

// Get the display name
String shortName = tz.getDisplayName(tz.inDaylightTime(today),
TimeZone.SHORT);
String longName = tz.getDisplayName(tz.inDaylightTime(today),
TimeZone.LONG);

// Get the number of hours from GMT
int rawOffset = tz.getRawOffset();
int hour = rawOffset / (60 * 60 * 1000);
int min = Math.abs(rawOffset / (60 * 1000)) % 60;

// Does the time zone have a daylight savings time period?
boolean hasDST = tz.useDaylightTime();

// Is the time zone currently in a daylight savings time?
boolean inDST = tz.inDaylightTime(today);

out.println(zoneIds[i] + ", " + shortName + ", " + longName + ", "
+ hour + ", " + min + ", hasDST: " + hasDST
+ ", inDST: " + inDST);
}
}

public static void main(String[] args) {

new TimeZoneAvailable().execute();
}
}







4 comentarios:

Daniel dijo...

Mae, es increible como miles de veces casi literalmente uno se mete en contratiempos por las fechas.
Esta muy bueno el toke.
Y si, porque lo ponen deprecated y ent no lo cambian por una version nueva... si lo van a seguir dejando ent quitenle el deprecated.

jsanca dijo...

Me parece que es fuertemente recomendado utilizar en los nuevos desarrollo esta librería: http://joda-time.sourceforge.net/, parece que la Java Community estaba estudiando la posibilidad de meter esta biblioteca en el JDK, pues el manejo de fechas es mucho mejor, que la pobre implementación actual.

jsanca dijo...

Este código tiene un bug, en el caso que se le pase una fecha con diferente año o mes, el resultado no es lo esperado, a continuación el código fixeado:

public static Date convertToTimeZoneDate(Date date, TimeZone timeZone) {

Date newTimeZoneDate = null;
Calendar foreignCalendar = null;
// Create a Calendar object with the local time zone
Calendar localCalendar = new GregorianCalendar();
localCalendar.setTime(date);

// Create an instance using foreign time zone and set it with the local
// UTC
foreignCalendar = new GregorianCalendar(timeZone);
foreignCalendar.set(Calendar.YEAR, localCalendar.get(Calendar.YEAR));
foreignCalendar.set(Calendar.DAY_OF_YEAR, localCalendar
.get(Calendar.DAY_OF_YEAR));
foreignCalendar.set(Calendar.HOUR_OF_DAY, localCalendar
.get(Calendar.HOUR_OF_DAY));
foreignCalendar
.set(Calendar.MINUTE, localCalendar.get(Calendar.MINUTE));
foreignCalendar
.set(Calendar.SECOND, localCalendar.get(Calendar.SECOND));

// Create a Calendar object with the local time zone and set
// the UTC from foreign calendar
newTimeZoneDate = new Date(foreignCalendar.getTimeInMillis());

return newTimeZoneDate;
} // convertToTimeZone.

kaleemsagard dijo...

Gracias por compartir el código. Me ha sido de gran utilidad. El API para fechas de Java es de lo peor que tiene la plataforma. Afortunadamente, y según he leído, joda se integrará en Java 8 para de una vez por todas tener una buena API de fechas.

Un saludo y nuevamente gracias