Oracle SQL:日期字段上的 ORA-01858 错误

时间:2021-07-06 10:53:56

标签: sql oracle oracle-sqldeveloper

我有很多名字中带有日期的表。例如MY_TABLE_2021_06_01, MY_TABLE_2021_06_02, etc。我试图从表名中提取日期,看看是否有任何表超过一年。这是我的代码:

select * from (
  select
      table_name,
to_date(substr(table_name,-2,2)||'/'||substr(table_name,-5,2)||'/'||substr(table_name,-10,4),'DD/MM/YYYY') TABLE_DATE
  from
      all_tables
  where
      owner = 'my_schema'
      and table_name like '%_20%'
)
where
    TABLE_DATE < trunc(sysdate)-365;

如果我不包含 where 子句 where table_date < trunc(sysdate)-365,上面的代码工作正常。如果我使用 where 子句运行代码,那么我会得到

<块引用>

ORA-01858: 在需要数字的地方发现了一个非数字字符 01858. 00000 - “在需要数字的地方发现了一个非数字字符” *原因:使用日期格式模型转换的输入数据是 不正确。输入数据不包含数字,其中数字是 格式模型所要求的。 *操作:修复输入数据或日期格式模型以确保 元素在数量和类型上匹配。然后重试操作。

我尝试了另一种冗长的方法,看看我是否可以让我的代码工作,但它仍然抛出相同的错误。我的新代码:

select * from (    
  select table_name, case when table_date < old_date then 1 else 0 end as OLD_TABLE from (
    select
        table_name,
    to_date(substr(table_name,-2,2)||'/'||substr(table_name,-5,2)||'/'||substr(table_name,-10,4),'DD/MM/YYYY') TABLE_DATE,
        trunc(sysdate)-365 OLD_DATE
    from
        all_tables
    where
        owner = 'my_schema'
        and table_name like '%_20%'
    )
 )
where 
    old_table = 1
order by
    old_table desc;

如果我不包含 where old_table = 1 子句,上面的代码也能正常工作,但在包含 where 子句时会出现相同的 ORA-01858 错误。我不明白这一点,因为字段 OLD_TABLE 不是日期字段,但仍然出现日期格式错误。

2 个答案:

答案 0 :(得分:1)

并非所有满足这些条件的表:

      owner = 'my_schema'
  and table_name like '%_20%'

具有“有效”的日期格式,或者它与您使用的格式掩码不匹配。

注意:如果表名是MY_TABLE_2021_06_01,你为什么要TO_DATE(..., 'DD/MM/YYYY')

2021_06_01 当然不是 DD/MM/YYYY 格式,而是 YYYY_MM_DDYYYY_DD_MM(不知道,06 可以是月或日; 01 也是如此)。 也许一旦你解决了这个问题,你的代码就会开始工作。

如果不是,则列出名称类似于 %_20% 的所有表,并查看其中哪些违反了您设置的规则。


对我有用的示例:首先是示例表:

SQL> CREATE TABLE my_table_2021_06_01
  2  (
  3     id   NUMBER
  4  );

Table created.

SQL> CREATE TABLE some_other_table_2020_02_17
  2  (
  3     id   NUMBER
  4  );

Table created.

查询以提取“名称”早于 1 年的表:

SQL> WITH
  2     tables
  3     AS
  4        (SELECT table_name,
  5                TO_DATE (REGEXP_SUBSTR (table_name, '\d+_\d+_\d+'),
  6                         'yyyy_mm_dd') datum
  7           FROM all_tables
  8          WHERE     owner = 'SCOTT'
  9                AND table_name LIKE '%20%')
 10  SELECT table_name
 11    FROM tables
 12   WHERE datum <= ADD_MONTHS (TRUNC (SYSDATE), -12);

TABLE_NAME
------------------------------
SOME_OTHER_TABLE_2020_02_17

SQL>

答案 1 :(得分:0)

我会使用这个逻辑来提取日期:

select to_date(substr(table_name, -10) default null on conversion error, 'YYYY_MM_DD')

这应该适用于与 sysdate 的比较。

这既可以处理与格式不匹配的表名,也可以简化日期算法。

相关问题