首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Oracle Create formula formula防止被零除

Oracle Create formula formula防止被零除
EN

Stack Overflow用户
提问于 2021-07-31 02:19:30
回答 3查看 69关注 0票数 0

我有一个包含公式列的表,其中有公式可以提取到动态SQL中,比如n5/n14+n3

我的问题是,当公式有除法,我应该尝试转换公式,以防止错误的divide by zero

我现在的想法是将公式转换为CASE,例如将n5/n14+n3转换为CASE WHEN n14 = 0 THEN 0 ELSE n5/n14+n3 END

但我只能在公式只有1个除法的情况下进行转换,不能在公式大于1除法的情况下进行转换(例如,n5/n4+n3/n2,或者公式有除法表达式(例如,n5/(n4+n3)

有人能解决这个问题吗?

代码语言:javascript
复制
WITH tmp AS 
(
    SELECT 'n5/n14+n3' AS formula FROM dual UNION ALL 
    SELECT 'n5/n14+n3/n1-n2' AS formula FROM dual UNION ALL 
    SELECT 'n8/(n17*n6)-n5/n4+n3/n1-n2' AS formula FROM dual UNION ALL 
    SELECT 'n5/(n14+n3*n2)-n1' AS formula FROM dual 
)
SELECT
    formula,
     CASE
         WHEN formula NOT LIKE '%/%' OR REPLACE(formula, ' ', '') LIKE '%/(%' OR (LENGTH(formula) - LENGTH(REPLACE(formula, '/', ''))) > 1 THEN  formula
         ELSE 'CASE WHEN ' || SUBSTR(REGEXP_SUBSTR(REPLACE(formula, ' ', ''), '/(n\d+)'), 2) || ' = 0 THEN 0 ELSE ' || formula || ' END'
     END AS formula1,
     'CASE WHEN ' || SUBSTR(REGEXP_SUBSTR(REPLACE(formula, ' ', ''), '/(n\d+)'), 2) || ' = 0 THEN 0 ELSE ' || formula || ' END' AS formula2
 FROM tmp t;

当前结果不是预期输出(仅第1行解决除以零错误):

代码语言:javascript
复制
formula                     formula1                                    formula2
n5/n14+n3                   CASE WHEN n14 = 0 THEN 0 ELSE n5/n14+n3 END CASE WHEN n14 = 0 THEN 0 ELSE n5/n14+n3 END                
n5/n14+n3/n1-n2             n5/n14+n3/n1-n2                             CASE WHEN n14 = 0 THEN 0 ELSE n5/n14+n3/n1-n2 END          
n8/(n17*n6)-n5/n4+n3/n1-n2  n8/(n17*n6)-n5/n4+n3/n1-n2                  CASE WHEN n4 = 0 THEN 0 ELSE n8/(n17*n6)-n5/n4+n3/n1-n2 END
n5/(n14+n3*n2)-n1           n5/(n14+n3*n2)-n1                           CASE WHEN  = 0 THEN 0 ELSE n5/(n14+n3*n2)-n1 END           
EN

回答 3

Stack Overflow用户

发布于 2021-07-31 02:22:31

您可以使用NULLIF()来防止错误。结果将是NULL

代码语言:javascript
复制
(
    SELECT 'n5/nullif(n14, 0)+n3' AS formula FROM dual UNION ALL 
    SELECT 'n5/nullif(n14, 0)+n3/nullif(n1, 0)-n2' AS formula FROM dual UNION ALL 
    SELECT 'n8/nullif(n17*n6, 0)-n5/nullif(n4, 0)+n3/nullif(n1, 0)-n2' AS formula FROM dual UNION ALL 
    SELECT 'n5/nullif((n14+n3*n2, 0)-n1' AS formula FROM dual 
)
票数 2
EN

Stack Overflow用户

发布于 2021-07-31 03:11:19

感谢@Linoff先生的建议,我确实使用了来获取和转换使用NULLIF的公式,如下所示

代码语言:javascript
复制
WITH tmp AS 
(
    SELECT 'n5/n14+n3' AS formula FROM dual UNION ALL 
    SELECT 'n5/n14+n3/n1-n2' AS formula FROM dual UNION ALL 
    SELECT 'n8/(n17*n6)-n5/n4+n3/n1-n2' AS formula FROM dual UNION ALL 
    SELECT 'n5/(n14+n3*n2)-n1' AS formula FROM dual 
)
SELECT formula,
    REGEXP_REPLACE(REPLACE(formula, ' ', ''), '\/(n\d+)|\/(\([^\)]*\))', '/NULLIF(\1\2,0)') AS formula1
FROM tmp;

结果:

代码语言:javascript
复制
formula                     formula1                                    
n5/n14+n3                   n5/NULLIF(n14,0)+n3                                     
n5/n14+n3/n1-n2             n5/NULLIF(n14,0)+n3/NULLIF(n1,0)-n2                     
n8/(n17*n6)-n5/n4+n3/n1-n2  n8/NULLIF((n17*n6),0)-n5/NULLIF(n4,0)+n3/NULLIF(n1,0)-n2
n5/(n14+n3*n2)-n1           n5/NULLIF((n14+n3*n2),0)-n1                             
票数 1
EN

Stack Overflow用户

发布于 2021-07-31 09:47:06

  • 用Java语言编写解析器以解析表达式(或使用现有解析器)。
  • 编写静态Java函数将解析器生成的抽象语法树转换为所需的输出,并使用NULLIFCASE表达式将除法中的每个分母包装起来,以防止引发被零除的异常。
  • 使用loadjava utility.
  • Write a PL/SQL函数将其加载到数据库中,以包装Java function.
  • Use PL/SQL函数,从而将动态代码转换为更安全的代码。

< code >F214

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/68598595

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档