我有很多名字中带有日期的表。例如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
不是日期字段,但仍然出现日期格式错误。
答案 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_DD
或 YYYY_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
的比较。
这既可以处理与格式不匹配的表名,也可以简化日期算法。