我得到200个供应商名称作为输入参数,存储过程具有巨大的值。我尝试在CLOB,VARCHAR2(32767)中声明。但是当我尝试执行它时,它不接受。 请有人帮我解决这个问题。
CREATE OR REPLACE PROCEDURE GMMT_PROC.or_rmce_RB_grid_sp (
p_plant_cd IN VARCHAR2,
p_region_cd IN VARCHAR2,
p_matrl_nbr IN OUT VARCHAR2,
p_supplier_nbr IN OUT VARCHAR2,
p_supplier_name IN OUT CLOB,
p_mrpcontrollercd IN OUT VARCHAR2,
p_currency IN VARCHAR2,
oresultset OUT sys_refcursor,
p_err_cd OUT NUMBER,
p_err_msg OUT VARCHAR2
)
IS
sqlquery varchar2(10000);
p1 varchar2(10);
p2 varchar2(20);
p3 varchar2(20);
p4 varchar2(20);
v_percent varchar2(10):='%';
V_FROM_CURRENCY VARCHAR2(5);
V_EX_RATE NUMBER;
v_matrl_nbr VARCHAR2(10000);
v_supplier_nbr VARCHAR2(10000);
v_supplier_name CLOB;
v_mrpcontrollercd VARCHAR2(10000);
BEGIN
p1:='matrl_nbr';
v_matrl_nbr:=p_matrl_nbr;
p2:='supplier_nbr';
v_supplier_nbr :=p_supplier_nbr;
p3:='supplier_name_txt';
v_supplier_name :=p_supplier_name;
p4:='mrp_controller_cd';
v_mrpcontrollercd:=p_mrpcontrollercd;
sqlquery:='select wk_nbr, nbr_working_days,'||p1||' as matrl_nbr ,'||p2||' as supplier_nbr,'||p3||' as supplier_name_txt,'||p4||' as mrp_controller_cd,sum(reqmnt_qty)reqmnt_qty , sum(proj_inven_qty)proj_inven_qty, sum(doh) doh,sum(proj_recpt_1) proj_recpt_1, sum(proj_recpt_2) proj_recpt_2, sum(proj_recpt_3) proj_recpt_3, sum(proj_recpt_4) proj_recpt_4,sum( proj_recpt_total_qty) proj_recpt_total_qty,sum(consumption_qty) consumption_qty, sum(consumption_var_qty) consumption_var_qty , sum(final_recpt_qty) final_recpt_qty,sum(recpt_var_qty) recpt_var_qty from GMMT_OWNER.OR_RMCE_GTT_PROJ_RECPT_T ';
sqlquery:=sqlquery||' where ('''||v_matrl_nbr||''' IS NULL) OR matrl_nbr IN (SELECT * FROM TABLE(GMMT_PROC.OR_in_list_RPT5_FN('''||v_matrl_nbr||'''))) OR matrl_nbr like '''|| v_matrl_nbr||v_percent||'''';
sqlquery:=sqlquery||' AND ('''||v_supplier_nbr||''' IS NULL) OR supplier_nbr IN (SELECT * FROM TABLE(GMMT_PROC.OR_in_list_RPT5_FN('''||v_supplier_nbr||'''))) OR supplier_nbr like '''|| v_supplier_nbr||v_percent||'''';
sqlquery:=sqlquery||' AND ('''||v_supplier_name||''' IS NULL) OR supplier_name_txt IN (SELECT * FROM TABLE(GMMT_PROC.OR_in_list_RPT5_FN('''||v_supplier_name||'''))) OR supplier_name_txt like '''|| v_supplier_name||v_percent||'''';
sqlquery:=sqlquery||' AND ('''||v_mrpcontrollercd||''' IS NULL) OR mrp_controller_cd IN (SELECT * FROM TABLE(GMMT_PROC.OR_in_list_RPT5_FN('''||v_mrpcontrollercd||'''))) OR mrp_controller_cd like '''|| v_mrpcontrollercd||v_percent||'''';
sqlquery:=sqlquery||' group by wk_nbr, nbr_working_days,'||p1||','||p2||','||p3||','||p4||' order by wk_nbr';
OPEN oresultset FOR sqlquery;
dbms_output.put_line(sqlquery);
END;
答案 0 :(得分:1)
您需要了解绑定变量!从不,永远连接传递给SQL语句的值。
请参阅http://docs.oracle.com/cd/E14072_01/appdev.112/e10472/dynamic.htm#CHDFCHHJ了解这是一个坏主意的原因。
现在你遇到的问题是由于将大值插入到SQL语句中,使得非常长字符串 - 对于varchar来说太长了。
有关如何将绑定变量与PL / SQL中的动态SQL一起使用,请参阅http://docs.oracle.com/cd/E14072_01/appdev.112/e10472/dynamic.htm#BJEDAHEE。
答案 1 :(得分:1)
动态SQL:不要这样做。
永远不要,永远不要将包含值的变量连接到SQL。 Never, please它会导致性能下降,难以重现/发现错误,并且对SQL注入非常开放。
可以使用动态SQL进行动态分组,因为P1 ... P4不是值,而是列名,因此无法绑定。
ALL other variables should be passed as binds:
你的OPEN声明应该是:
OPEN oresultset FOR '
SELECT wk_nbr,
nbr_working_days,
' || dbms_assert.simple_sql_name(p1) || ' AS matrl_nbr,
' || dbms_assert.simple_sql_name(p2) || ' AS supplier_nbr,
' || dbms_assert.simple_sql_name(p3) || ' AS supplier_name_txt,
' || dbms_assert.simple_sql_name(p4) || ' AS mrp_controller_cd,
sum(reqmnt_qty) reqmnt_qty,
[...]
FROM GMMT_OWNER.OR_RMCE_GTT_PROJ_RECPT_T
WHERE (:VAR1 IS NULL)
OR matrl_nbr IN (SELECT * FROM TABLE(GMMT_PROC.OR_in_list_RPT5_FN(:VAR1)))
OR matrl_nbr LIKE :VAR1 || ''%''
AND (:VAR2 IS NULL)
OR supplier_nbr IN (SELECT * FROM TABLE(GMMT_PROC.OR_in_list_RPT5_FN(:VAR2)))
OR supplier_nbr LIKE :VAR2 || ''%''
[...]
GROUP BY wk_nbr,
nbr_working_days,
' || p1 || ', ' || p2 || ', ' || p3 || ', ' || p4 || '
ORDER BY wk_nbr'
USING v_matrl_nbr, v_matrl_nbr, v_matrl_nbr,
v_supplier_nbr, v_supplier_nbr, v_supplier_nbr,
[...];
为防止用户滥用此动态SQL文件,您必须确保p1 ... p4已从您定义的列表中预先批准,或使用上述DBMS_ASSERT。
实际上,如果列表很小,您可以使用简单的静态SQL:
OPEN cc FOR
SELECT wk_nbr,
nbr_working_days,
decode(p1, 'column1', column1, 'column2', column2 [...]) AS matrl_nbr,
[...]
GROUP BY [...]
decode(p1, 'column1', column1, 'column2', column2 [...]),
[...];