2015-11-04 42 views
1

我在过去的两周中一直受到这个问题的折磨,无法再花费更多时间。如果你有一分钟​​的话,我很乐意听到这个社区的任何想法。我是自学成才的,所以我必须按照内部条件解释这一点。谢谢您的考虑。Oracle SQL:使用存储过程获取由逗号分隔的ID字段连接的数据

我的目标是获取所有非零余额的发票。这对于定期发票来说足够简单。问题出现在付款提醒中,因为它们的余额只是滞纳金,所以为了取得总余额,我必须通过连接表COLLECTIVEINVOICENO中的单个字段链接回原始发票。表INVOICE

例子:

invoiceid  | invoiceno  | invoicetypename  | amount 
--------------|---------------|---------------------|-------------------- 
10008   | 123000  | Payment Reminder | 5 
10005   | 113000  | Payment Reminder | 5 
10001   | 110000  | Invoice    | 35 

COLLECTIVEINVOICENO其中1000110005过期的例子:

invoiceid  | followed_up_in (CHAR) 
--------------|---------------------------------------------------------- 
10005   | 113000, 123000 
10001   | 113000 

我已经找到了一个非常坏的解决方案,其工作方式获取的总账一次只有一个客户,但它对资源的要求太高,我无法在整个桌子上使用它INVOICE

select (
    select sum(i.amount) as sum 
    from INVOICE i0 
    join COLLECTIVEINVOICENO cin0 on cin0.invoiceid = i0.invoiceid 
    where instr(cin0.followed_up_in, i.invoiceno) > 0 
) as original_sum 
from INVOICE i 

这全表扫描,重复5次在我的主select以各种余额为不同的列进行排序,并取出来自原始发票等资料,除了他们的平衡,比如它们的位置等

我最初的意图是将子查询左键加入INVOICE i作为视图,但Oracle在instr(cin0.collectiveinvoiceno, i.invoice)(ORA-00904:“I”,“INVOICENO”:无效标识符)中不识别i.invoice

我现在正在寻找目前正在做的游标等,但我真的想知道是否有任何SQL,非PL的方式呢?

谢谢。

编辑:关于将逗号分隔的数据分成行,即。改变数据模型以促进关系:这肯定会帮助我确信,但是数据模型与第三方软件打包在一起,所以我对此无能为力。 (如果他们确实在某些时候改变了它,我不会感到惊讶)

编辑2:尝试将逗号分隔数据拆分为行,使用regexp_substr()levelconnect by并将此视图加入主表。我没有解决问题,因为当regexp_substr()在其他表加入时资源明显昂贵,并且即使在半小时后查询也无法提供。谷歌显示,这是正则表达式的常见问题,建议您尝试使用存储过程。

我已经改变了标题以反映这一点,我将不胜感激关于如何使用存储过程来解决这个问题的任何输入。不管怎样,谢谢!

+0

[字符串分割为多行中的Oracle]的可能的复制(http://stackoverflow.com/questions/14328621/splitting-string-into-multiple-rows-in-oracle) – gmiley

+5

存储逗号分隔关系列中的值是通向心痛的路径。你确定你不想修复底层的数据模型问题,这样你就可以让数据库为你工作,而不是让它与你作斗争? –

+0

我不确定您是否错误地键入了您的示例,或者我只是没有遵循数据模型。根据你在这里给出的样本数据,你是描述,很难判断10001和10005是否独立。你期望看到什么最终的平衡?只有一行 - INVOICE 10001 - 余额为45? – KevinKirkpatrick

回答

1

这可能有帮助。它将分隔逗号分隔followed_up_in,并在加入主表之前使用xmltable将产生的列表值平铺成发票号码行。

 select i0.invoiceid, sum(i0.amount) 
     from INVOICE i0 
     left outer join 
     (select invoice_id, trim(COLUMN_VALUE) invoiceno 
      FROM COLLECTIVEINVOICENO, 
       xmltable(('"' || REPLACE(followed_up_in, ',', '","') || '"')) 
    ) cin0 
     on cin0.invoiceno = i0.invoiceno 
     and i0.invoiceid = cin0.invoiceid 
     group by i0.invoiceid 
+0

谢谢Thomas,看起来很棒。我明天会尝试。 – IngridSkard

+0

我遵循了你的建议托马斯和甲骨文运行它,但它运行“永远”。当单元测试时,我发现regexp_substr()似乎会导致挂断。当我从dual运行regexp_substr()时,它的运行没有延迟。我不知道开发者是否可以对他们的观点(我正在从观点来看)使用'level'这个限制,或者是这样。不管怎么说,还是要谢谢你。 – IngridSkard

+0

呃,事实证明'regexp_substr()'只能在主表上没有其他东西被加入时传递,即在视图中和主'select'下! – IngridSkard