2

我继承了一个 Oracle .dmp 文件,我正在尝试将其导入 CSV,以便将其加载到 MySQL 中。

这里描述了我使用的一般方法。不过,我的一排有问题。它包含 5544-09-14 的日期,如下所示:

alter session set nls_date_format = 'dd-MON-yyyy';
select OID, REF, TRADING_DATE From LOAN WHERE REF = 'XXXX';

OID REF                  TRADING_DATE
--- -------------------- ------------
1523 XXXX                 14-SEP-5544

这是来自未验证输入日期的遗留系统的垃圾数据。我想知道为什么我的 PL/SQL 函数导出数据会阻塞这个值?

它以“0000-00-00T00:00:00”的 TRADING_DATE 值导出该行,我不知道为什么?

SELECT dump(TRADING_DATE) FROM LOAN WHERE REF = 'XXXX';

DUMP(TRADING_DATE)
--------------------------------------------------------------------------------
Typ=12 Len=7: 44,156,9,14,1,1,1

SELECT to_char(trading_date, 'YYYYMMDDHH24MISS') FROM LOAN WHERE REF = 'XXXX';
TO_CHAR(TRADIN
--------------
00000000000000
4

1 回答 1

8

该列中存储的值不是有效日期。的第一个字节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。和这个前面的问题有些相关。

于 2012-09-17T09:13:49.287 回答