Oracle基本数据类型存储格式浅析——日期类型TIMESTAMP WITH TIME ZONE
原文链接:https://www.modb.pro/db/22143?cyn
摘要:描述TIMESTAMP WITH TIME ZONE以及TIMESTAMP WITH LOCAL TIME ZONE类型的存储结构本文对TIMESTAMP WITH LOCAL TIME ZONE和TIMESTAMP WITH TIME ZONE类型的存储格式进行简单的说明。
SQL>CREATETABLETEST_TIMESTAMP(TIME1TIMESTAMP(9),TIME2TIMESTAMP(6)WITHLOCALTIMEZONE,2TIME3TIMESTAMP(4)WITHTIMEZONE);表已创建。SQL>INSERTINTOTEST_TIMESTAMPVALUES(SYSTIMESTAMP,SYSTIMESTAMP,SYSTIMESTAMP);已创建1行。SQL>SELECT*FROMTEST_TIMESTAMP;TIME1----------------------------------------------------TIME2----------------------------------------------------TIME3----------------------------------------------------11-1月-0511.08.15.027000000下午11-1月-0511.08.15.027000下午11-1月-0511.08.15.0270下午+08:00SQL>SELECTDUMP(TIME1,16),DUMP(TIME2,16),DUMP(TIME3,16)FROMTEST_TIMESTAMP;DUMP(TIME1,16)-------------------------------------------------------------DUMP(TIME2,16)-------------------------------------------------------------DUMP(TIME3,16)-------------------------------------------------------------Typ=180Len=11:78,69,1,b,18,9,10,1,9b,fc,c0Typ=231Len=11:78,69,1,b,18,9,10,1,9b,fc,c0Typ=181Len=13:78,69,1,b,10,9,10,1,9b,fc,c0,1c,3c
可以发现,如果客户端和数据库中的时区是一致的,那么TIMESTAMP和TIMESTAMP WITH LOCAL TIME ZONE存储的数据是完全一样的。
TIMESTAMP WITH TIME ZONE则略有不同,它保存的是0时区的时间,和所处的时区信息。
修改客户端主机的时区,由东8区(+8区)改为0时区。
SQL>INSERTINTOTEST_TIMESTAMPVALUES(SYSTIMESTAMP,SYSTIMESTAMP,SYSTIMESTAMP);已创建1行。
修改客户端主机的时区,改为西5区(-5时区)。
SQL>INSERTINTOTEST_TIMESTAMPVALUES(SYSTIMESTAMP,SYSTIMESTAMP,SYSTIMESTAMP);已创建1行。
修改客户端主机的时区,改为西12区(-12时区)。
SQL>INSERTINTOTEST_TIMESTAMPVALUES(SYSTIMESTAMP,SYSTIMESTAMP,SYSTIMESTAMP);已创建1行。
修改客户端主机的时区,改为东13区(+13时区)。
SQL>INSERTINTOTEST_TIMESTAMPVALUES(SYSTIMESTAMP,SYSTIMESTAMP,SYSTIMESTAMP);已创建1行。
修改客户端主机的时区,改为西3.5区(-3.5时区)。
SQL>INSERTINTOTEST_TIMESTAMPVALUES(SYSTIMESTAMP,SYSTIMESTAMP,SYSTIMESTAMP);已创建1行。
修改客户端主机的时区,改为东9.5区(+9.5时区)。
SQL>INSERTINTOTEST_TIMESTAMPVALUES(SYSTIMESTAMP,SYSTIMESTAMP,SYSTIMESTAMP);已创建1行。SQL>COMMIT;提交完成。
修改客户端主机的时区,改回东8区(+8时区)。
SQL>SELECT*FROMTEST_TIMESTAMP;TIME1-----------------------------------------------TIME2-----------------------------------------------TIME3-----------------------------------------------11-1月-0511.08.15.027000000下午11-1月-0511.08.15.027000下午11-1月-0511.08.15.0270下午+08:0011-1月-0503.11.43.746000000下午11-1月-0511.11.43.746000下午11-1月-0503.11.43.7460下午+00:0011-1月-0510.14.08.987000000上午11-1月-0511.14.08.987000下午11-1月-0510.14.08.9870上午-05:0011-1月-0503.15.01.732000000上午11-1月-0511.15.01.732000下午11-1月-0503.15.01.7320上午-12:0012-1月-0504.20.21.522000000上午11-1月-0511.20.21.522000下午12-1月-0504.20.21.5220上午+13:0011-1月-0502.15.16.567000000下午12-1月-0501.45.16.567000上午11-1月-0502.15.16.5670下午-03:3012-1月-0503.16.54.992000000上午12-1月-0501.46.54.992000上午12-1月-0503.16.54.9920上午+09:30已选择7行。SQL>SELECTDUMP(TIME1,16),DUMP(TIME2,16),DUMP(TIME3,16)FROMTEST_TIMESTAMP;DUMP(TIME1,16)-------------------------------------------------------------DUMP(TIME2,16)-------------------------------------------------------------DUMP(TIME3,16)-------------------------------------------------------------Typ=180Len=11:78,69,1,b,18,9,10,1,9b,fc,c0Typ=231Len=11:78,69,1,b,18,9,10,1,9b,fc,c0Typ=181Len=13:78,69,1,b,10,9,10,1,9b,fc,c0,1c,3cTyp=180Len=11:78,69,1,b,10,c,2c,2c,77,e,80Typ=231Len=11:78,69,1,b,18,c,2c,2c,77,e,80Typ=181Len=13:78,69,1,b,10,c,2c,2c,77,e,80,14,3cTyp=180Len=11:78,69,1,b,b,f,9,3a,d4,6c,c0Typ=231Len=11:78,69,1,b,18,f,9,3a,d4,6c,c0Typ=181Len=13:78,69,1,b,10,f,9,3a,d4,6c,c0,f,3cTyp=180Len=11:78,69,1,b,4,10,2,2b,a1,6f,0Typ=231Len=11:78,69,1,b,18,10,2,2b,a1,6f,0Typ=181Len=13:78,69,1,b,10,10,2,2b,a1,6f,0,8,3cTyp=180Len=11:78,69,1,c,5,15,16,1f,1d,16,80Typ=231Len=11:78,69,1,b,18,15,16,1f,1d,16,80Typ=181Len=13:78,69,1,b,10,15,16,1f,1d,16,80,21,3cTyp=180Len=11:78,69,1,b,f,10,11,21,cb,bb,c0Typ=231Len=11:78,69,1,c,2,2e,11,21,cb,bb,c0Typ=181Len=13:78,69,1,b,12,2e,11,21,cb,bb,c0,11,1eTyp=180Len=11:78,69,1,c,4,11,37,3b,20,b8,0Typ=231Len=11:78,69,1,c,2,2f,37,3b,20,b8,0Typ=181Len=13:78,69,1,b,12,2f,37,3b,20,b8,0,1d,5aSQL>SELECTTO_NUMBER('1C','XXX'),TO_NUMBER('3C','XXX')FROMDUAL;TO_NUMBER('1C','XXX')TO_NUMBER('3C','XXX')------------------------------------------2860SQL>SELECTTO_NUMBER('14','XXX'),TO_NUMBER('3C','XXX'),TO_NUMBER('143C','XXXXXXX')FROMDUAL;TO_NUMBER('14','XXX')TO_NUMBER('3C','XXX')------------------------------------------2060SQL>SELECTTO_NUMBER('3C','XXX'),TO_NUMBER('1E','XXX'),TO_NUMBER('5A','XXX')FROMDUAL;TO_NUMBER('3C','XXX')TO_NUMBER('1E','XXX')TO_NUMBER('5A','XXX')---------------------------------------------------------------603090
可以看出,修改时区会导致系统TIMESTAMP时间发生变化,但是对于TIMESTAMP WITH LOCAL TIME ZONE类型,总是将系统的时间转化到数据库服务器上时区的时间进行存储。
TIMESTAMP WITH TIME ZONE保存的是当前时间转化到0时区的对应的时间,并通过最后两位来保存时区信息。
第一位表示时区的小时部分。0时区用0x14表示。东n区在这个基础上加n,西n区在这个基础上减n。我们所处的东8区表示为0x1C。西5区表示为0xF。
第二位表示时区的分钟部分。标准是0x3C,即60分钟。对于东时区的半区,在这个基础上加上30分钟,如果是西时区,则减去30分钟。
更多Oracle经典案例解析:https://www.modb.pro/u/379?cyn
声明:本站所有文章资源内容,如无特殊说明或标注,均为采集网络资源。如若本站内容侵犯了原著者的合法权益,可联系本站删除。