该列中存储的值不是有效日期。的第一个字节dump
应该是世纪,根据 Oracle 支持说明 69028.1,它以“excess-100”表示法存储,这意味着它的值应该是 100 + 实际世纪;所以1900是119,2000是120,5500是155。所以44代表-5600;您存储的日期似乎实际上代表 5544-09-14 BC。由于 Oracle 仅支持年份介于 -4713 和 +9999 之间的日期,因此无法识别。
你可以很容易地重新创建它;最棘手的一点是首先将无效日期放入数据库:
create table t42(dt date);
Table created.
declare
d date;
begin
dbms_stats.convert_raw_value('2c9c090e010101', d);
insert into t42 (dt) values (d);
end;
/
PL/SQL procedure successfully completed.
select dump(dt), dump(dt, 1016) from t42;
DUMP(DT)
--------------------------------------------------------------------------------
DUMP(DT,1016)
--------------------------------------------------------------------------------
Typ=12 Len=7: 45,56,9,14,1,1,1
Typ=12 Len=7: 2d,38,9,e,1,1,1
因此,这有一行与您所做的数据相同。使用alter session
我可以看到看起来像有效日期的内容:
alter session set nls_date_format = 'DD-Mon-YYYY';
select dt from t42;
DT
-----------
14-Sep-5544
alter session set nls_date_format = 'YYYYMMDDHH24MISS';
select dt from t42;
DT
--------------
55440914000000
但是,如果我使用明确的日期掩码,它只会得到零:
select to_char(dt, 'DD-Mon-YYYY'), to_char(dt, 'YYYYMMDDHH24MISS') from t42;
TO_CHAR(DT,'DD-MON-Y TO_CHAR(DT,'YY
-------------------- --------------
00-000-0000 00000000000000
如果我运行你的程序:
exec dump_table_to_csv('T42');
生成的 CSV 具有:
"DT"
"0000-00-00T00:00:00"
我认为不同之处在于那些试图显示日期的人坚持使用内部日期数据类型 12,而那些显示零的人使用外部数据类型 13,如注释 69028.1 中所述。
所以简而言之,你的程序没有做错任何事情,它试图导出的日期在内部是无效的。除非您知道它应该是什么日期,考虑到您的起点,这似乎不太可能,我认为除了猜测或忽略它之外,您无能为力。除非,也许,您知道数据是如何插入的,并且可以计算出它是如何被破坏的。
我认为它更有可能来自 OCI 程序,而不是我在这里所做的;这个“原始”技巧最初来自这里。您可能还想查看注释 331831.1。和这个前面的问题有些相关。