很多人都知道,在Oracle 9i之前,雖然有一個NEW_TIME函數可以改變DATE的時間戳部分,但是還沒有專門用來存儲時區信息的數據類型。在Oracle9i中,我們可以使用DBTIMEZONE僞字段查詢數據庫的時區,使用SESSIONTIMEZONE僞字段查詢會話的時區。
但是,對於大多數數據庫,這些值都是-07:00之類的偏移值,因此對於NEW_TIME函數是沒有用的。Oracle9i有關NEW_TIME的文檔建議使用FROM_TZ來替代,但是這可能會產生誤導。FROM_TZ只將一個時區應用到一個時間戳上;它並不能把一個時區轉換成另外一個時區。
其實有一個比較好的方法(從文檔中得到這個方法可能有點難)。首先,爲了完成這個工作,在正確的時區內需要一個TIMESTAMP WITH ZONE數據類型。然後,如果你將關鍵字AT TIME ZONE應用到那個值,它就會自動地調整爲新的時區和日期。
select (timestamp 2003-04-06 01:59:59 at time zone PDT)
at time zone GMT
from dual;
06-APR-03 08.59.59.00000000 AM GMT
這個語句將爲太平洋白天時間(即其切換到PST之前的時刻)構造一個TIMESTAMP WITH TIME ZONE然後再將其轉換到GMT。AT TIME ZONE關鍵字也接受默認的偏移值語法:
select (timestamp 2003-04-06 02:00:00
at time zone -07:00) at time zone
00:00 from dual;
06-APR-03 09.00.00.000000000 AM +00:00
你還可以使用僞字段來自動調整當前會話的時區:
selectcurrent_timestamp at time zone dbtimezone from dual;
上面的表達式返回一個當前會話的本地時間(數據類型爲時區),重新調整數據庫的時區,調整後的時區將與SYSTIMESTAMP的結果相等。
有了以上的這些信息,就可以構造一個比較好的NEW_TIME函數:
create or replace function my_new_time
(
p_dwtz timestamp with time zone,
p_tz varchar2
) return date
is
begin
return cast(p_dwtz at time zone p_tz as date);
end my_new_time;
/
show errors;
select my_new_time(sysdate,+08:00) from dual;
如果第一個參數被標記爲一個timestamp with time zone,你可以傳入一個TIMESTAMP和DATE,這樣由於Oracle的自動轉型操作,得到的時間將是會話在本地時區的當前時間。這個函數接受包括偏移值在內的任何可以被TIMESTAMP識別的時區,然後將接受的時區調整爲正確的值。
上海阜和進修學校