在进行国际性软件项目开发的过程中,有时候会碰到一些比较特殊的要求。比如:比如说,你做的是个购物网站(假设服务器放在中国上海),当全世界客户在你的网站上下订单买东西后,往往希望看到客户所在地下单时间,比如说我是个美国纽约人,我在你的网站上下单后,你给我看到一个上海的下单时间,会觉得非常的奇怪。众所周知,纽约时间相对上海时间大约要晚13小时,如果让客户看到本地时区的时间,将变得更加符合客户的时间观念,使得客户理解比较方便。
其实,java中早已考虑过世界时区(TimeZone)这个问题,并给出了比较合理的解决方法,可以比较方便的进行世界时区时间的转化,将一个时区的时间转换成另一个时区的时间。可以看看下面的的实际例子(运行例子的main()方法)。
关于如何知道客户所在的时区,可以根据客户所在的ip或者用户注册提供的国家来计算出所在的时区。
- /*
- * Created on 2005-6-10
- * Author stephen
- * Email zhoujianqiang AT gmail DOT com
- * CopyRight(C)2005-2008 , All rights reserved.
- */
- package com.soft4j.utility;
- import java.text.ParseException;
- import java.text.SimpleDateFormat;
- import java.util.Date;
- import java.util.GregorianCalendar;
- import java.util.TimeZone;
- import java.util.Vector;
- import com.soft4j.log.Log;
- /**
- * 与日期、时间相关的一些常用工具方法.
- * <p>
- * 日期(时间)的常用格式(formater)主要有: <br>
- * yyyy-MM-dd HH:mm:ss <br>
- *
- * @author stephen
- * @version 1.0.0
- */
- public final class DateTool {
- /**
- * 对日期(时间)中的日进行加减计算. <br>
- * 例子: <br>
- * 如果Date类型的d为 2005年8月20日,那么 <br>
- * calculateByDate(d,-10)的值为2005年8月10日 <br>
- * 而calculateByDate(d,+10)的值为2005年8月30日 <br>
- *
- * @param d
- * 日期(时间).
- * @param amount
- * 加减计算的幅度.+n=加n天;-n=减n天.
- * @return 计算后的日期(时间).
- */
- public static Date calculateByDate(Date d, int amount) {
- return calculate(d, GregorianCalendar.DATE, amount);
- }
- public static Date calculateByMinute(Date d, int amount) {
- return calculate(d, GregorianCalendar.MINUTE, amount);
- }
- public static Date calculateByYear(Date d, int amount) {
- return calculate(d, GregorianCalendar.YEAR, amount);
- }
- /**
- * 对日期(时间)中由field参数指定的日期成员进行加减计算. <br>
- * 例子: <br>
- * 如果Date类型的d为 2005年8月20日,那么 <br>
- * calculate(d,GregorianCalendar.YEAR,-10)的值为1995年8月20日 <br>
- * 而calculate(d,GregorianCalendar.YEAR,+10)的值为2015年8月20日 <br>
- *
- * @param d
- * 日期(时间).
- * @param field
- * 日期成员. <br>
- * 日期成员主要有: <br>
- * 年:GregorianCalendar.YEAR <br>
- * 月:GregorianCalendar.MONTH <br>
- * 日:GregorianCalendar.DATE <br>
- * 时:GregorianCalendar.HOUR <br>
- * 分:GregorianCalendar.MINUTE <br>
- * 秒:GregorianCalendar.SECOND <br>
- * 毫秒:GregorianCalendar.MILLISECOND <br>
- * @param amount
- * 加减计算的幅度.+n=加n个由参数field指定的日期成员值;-n=减n个由参数field代表的日期成员值.
- * @return 计算后的日期(时间).
- */
- private static Date calculate(Date d, int field, int amount) {
- if (d == null)
- return null;
- GregorianCalendar g = new GregorianCalendar();
- g.setGregorianChange(d);
- g.add(field, amount);
- return g.getTime();
- }
- /**
- * 日期(时间)转化为字符串.
- *
- * @param formater
- * 日期或时间的格式.
- * @param aDate
- * java.util.Date类的实例.
- * @return 日期转化后的字符串.
- */
- public static String date2String(String formater, Date aDate) {
- if (formater == null || "".equals(formater))
- return null;
- if (aDate == null)
- return null;
- return (new SimpleDateFormat(formater)).format(aDate);
- }
- /**
- * 当前日期(时间)转化为字符串.
- *
- * @param formater
- * 日期或时间的格式.
- * @return 日期转化后的字符串.
- */
- public static String date2String(String formater) {
- return date2String(formater, new Date());
- }
- /**
- * 获取当前日期对应的星期数.
- * <br>1=星期天,2=星期一,3=星期二,4=星期三,5=星期四,6=星期五,7=星期六
- * @return 当前日期对应的星期数
- */
- public static int dayOfWeek() {
- GregorianCalendar g = new GregorianCalendar();
- int ret = g.get(java.util.Calendar.DAY_OF_WEEK);
- g = null;
- return ret;
- }
- /**
- * 获取所有的时区编号. <br>
- * 排序规则:按照ASCII字符的正序进行排序. <br>
- * 排序时候忽略字符大小写.
- *
- * @return 所有的时区编号(时区编号已经按照字符[忽略大小写]排序).
- */
- public static String[] fecthAllTimeZoneIds() {
- Vector v = new Vector();
- String[] ids = TimeZone.getAvailableIDs();
- for (int i = 0; i < ids.length; i++) {
- v.add(ids[i]);
- }
- java.util.Collections.sort(v, String.CASE_INSENSITIVE_ORDER);
- v.copyInto(ids);
- v = null;
- return ids;
- }
- /**
- * 测试的main方法.
- *
- * @param argc
- */
- public static void main(String[] argc) {
- String[] ids = fecthAllTimeZoneIds();
- String nowDateTime =date2String("yyyy-MM-dd HH:mm:ss");
- System.out.println("The time Asia/Shanhai is " + nowDateTime);//程序本地运行所在时区为[Asia/Shanhai]
- //显示世界每个时区当前的实际时间
- for(int i=0;i <ids.length;i++){
- System.out.println(" * " + ids[i] + "=" + string2TimezoneDefault(nowDateTime,ids[i]));
- }
- //显示程序运行所在地的时区
- System.out.println("TimeZone.getDefault().getID()=" +TimeZone.getDefault().getID());
- }
- /**
- * 将日期时间字符串根据转换为指定时区的日期时间.
- *
- * @param srcFormater
- * 待转化的日期时间的格式.
- * @param srcDateTime
- * 待转化的日期时间.
- * @param dstFormater
- * 目标的日期时间的格式.
- * @param dstTimeZoneId
- * 目标的时区编号.
- *
- * @return 转化后的日期时间.
- */
- public static String string2Timezone(String srcFormater,
- String srcDateTime, String dstFormater, String dstTimeZoneId) {
- if (srcFormater == null || "".equals(srcFormater))
- return null;
- if (srcDateTime == null || "".equals(srcDateTime))
- return null;
- if (dstFormater == null || "".equals(dstFormater))
- return null;
- if (dstTimeZoneId == null || "".equals(dstTimeZoneId))
- return null;
- SimpleDateFormat sdf = new SimpleDateFormat(srcFormater);
- try {
- int diffTime = getDiffTimeZoneRawOffset(dstTimeZoneId);
- Date d = sdf.parse(srcDateTime);
- long nowTime = d.getTime();
- long newNowTime = nowTime - diffTime;
- d = new Date(newNowTime);
- return date2String(dstFormater, d);
- } catch (ParseException e) {
- Log.output(e.toString(), Log.STD_ERR);
- return null;
- } finally {
- sdf = null;
- }
- }
- /**
- * 获取系统当前默认时区与UTC的时间差.(单位:毫秒)
- *
- * @return 系统当前默认时区与UTC的时间差.(单位:毫秒)
- */
- private static int getDefaultTimeZoneRawOffset() {
- return TimeZone.getDefault().getRawOffset();
- }
- /**
- * 获取指定时区与UTC的时间差.(单位:毫秒)
- *
- * @param timeZoneId
- * 时区Id
- * @return 指定时区与UTC的时间差.(单位:毫秒)
- */
- private static int getTimeZoneRawOffset(String timeZoneId) {
- return TimeZone.getTimeZone(timeZoneId).getRawOffset();
- }
- /**
- * 获取系统当前默认时区与指定时区的时间差.(单位:毫秒)
- *
- * @param timeZoneId
- * 时区Id
- * @return 系统当前默认时区与指定时区的时间差.(单位:毫秒)
- */
- private static int getDiffTimeZoneRawOffset(String timeZoneId) {
- return TimeZone.getDefault().getRawOffset()
- - TimeZone.getTimeZone(timeZoneId).getRawOffset();
- }
- /**
- * 将日期时间字符串根据转换为指定时区的日期时间.
- *
- * @param srcDateTime
- * 待转化的日期时间.
- * @param dstTimeZoneId
- * 目标的时区编号.
- *
- * @return 转化后的日期时间.
- * @see #string2Timezone(String, String, String, String)
- */
- public static String string2TimezoneDefault(String srcDateTime,
- String dstTimeZoneId) {
- return string2Timezone("yyyy-MM-dd HH:mm:ss", srcDateTime,
- "yyyy-MM-dd HH:mm:ss", dstTimeZoneId);
- }
- }