游标_共享
The ON CONVERSION ERROR
当参数 CURSOR_SHARING 设置为 FORCE 时,该功能不起作用。要避免此错误,请更改系统、会话或语句级别的参数。
理想情况下,整个系统的 CURSOR_SHARING 应设置为 EXACT。但是如果我们有一个不使用绑定变量的应用程序,我们可能无法运行alter system set cursor_sharing=exact;
.
该参数可以在会话级别设置alter session set cursor_sharing=exact;
,但不断更改会话参数并不总是很方便。
可以使用提示在语句级别更改参数CURSOR_SHARING_EXACT
:
SQL> select /*+ cursor_sharing_exact */ to_date(the_date default null on conversion error, 'MM/DD/YYYY') the_date
2 from
3 (
4 select '1/1/2021' the_date from dual union all
5 select 'bad date' the_date from dual
6 );
THE_DATE
---------
01-JAN-21
解析器/优化器错误
正如 @gouessej 发现的那样,ORA-43918 错误还有另一个与游标共享无关的潜在原因。似乎存在与转换相关的解析或优化器错误CASE
and TO_
在某些版本的 Oracle 上运行。
例如,以下 SQL 语句在 Oracle 18c 和 19c 上失败:
SQL> select case when v_num is null then 0 else v_num end
2 from
3 (
4 select to_number('120.3' default null on conversion error, '99999D99') as v_num
5 from dual
6 );
select to_number('120.3' default null on conversion error, '99999D99') as v_num
*
ERROR at line 4:
ORA-43918: This argument must be a literal
我相信这是一个解析或优化器错误,因为如果您通过添加像这样的谓词来停止转换,错误就会消失rownum >= 1
。 (当 Oracle 看到ROWNUM
,它假设结果必须按特定顺序显示,并且不会对该查询块应用尽可能多的转换。)
SQL> select case when v_num is null then 0 else v_num end
2 from
3 (
4 select to_number('120.3' default null on conversion error, '99999D99') as v_num
5 from dual
6 where rownum >= 1
7 );
CASEWHENV_NUMISNULLTHEN0ELSEV_NUMEND
------------------------------------
120.3