由以下原因引起:java.sql.SQLException:ORA-06550:第 1 行,第 7 列:PLS-00306:调用'PR_SP_FAHMI'时参数的数量或类型错误



我有一个java文件,它用于通过使用SimpleJdbc来执行存储过程,如下面的代码所示:

@Transactional(propagation = Propagation.REQUIRED, rollbackFor = {SQLException.class, Exception.class })
public void executeSPForInsertData(DataSource ds,String procedureName,Map<String, Object> inputParameter){
//PARAMS.PKG_PARA_UPLD_VAL.PR_SP_FAHMI
String[] inParam = inputParameter.keySet().toArray(new String[0]);
SqlParameter[] in = new SqlParameter[inParam.length];
String[] buff = new String[inParam.length * 2];
SimpleJdbcCall jdbcCall = new SimpleJdbcCall(paramsDataSourceBean)
.withProcedureName(procedureName)
.withoutProcedureColumnMetaDataAccess()
.useInParameterNames(buff).declareParameters(in);
String[] path = procedureName.split(".");
if (path.length >= 2) {
jdbcCall.withSchemaName(path[0]);
jdbcCall.withCatalogName(path[1]);
jdbcCall.withProcedureName(path[2]);
}
Map<String,Object> outputParameter = jdbcCall.execute(inputParameter);
}

这是我的存储过程

PROCEDURE PR_SP_FAHMI (P_T_TABLE_UPLD_EXCEL IN PARAMS.EXCEL)
is 
P_LOGID VARCHAR2(255);
BEGIN
BEGIN
INSERT INTO PARAMS.EMPTY
SELECT 
C.PARA_OBJT_GROUP            ,
C.PARA_OBJT_CODE             ,
C.PARA_PROD_MATRIX_ID        ,
C.PARA_PROD_CHANNEL_ID       ,
C.PARA_PROD_SALES_GROUP_ID   ,
C.PARA_CUST_GROUP            ,
C.PARA_SLS_THROUGH_ID        ,
C.ACTIVE                     
FROM TABLE(P_T_TABLE_UPLD_EXCEL) C;
EXCEPTION
WHEN NO_DATA_FOUND THEN
NULL;
WHEN OTHERS THEN
RAISE_APPLICATION_ERROR(-20001, 'ERROR-' || SQLERRM);
END;
END PR_SP_FAHMI;

这是excel 申报类型

CREATE OR REPLACE TYPE EXCEL AS TABLE OF PARAMS.T_OBJECT_FROM_EXCEL

这是声明类型T_OBJECT_FROM_EXCEL

CREATE OR REPLACE TYPE "T_OBJECT_FROM_EXCEL" FORCE AS OBJECT (
para_objt_group            VARCHAR2(3),
para_objt_code             VARCHAR2(3),
para_prod_matrix_id        VARCHAR2(5),
para_prod_channel_id       VARCHAR2(3),
para_prod_sales_group_id   VARCHAR2(5),
para_cust_group            VARCHAR2(3),
para_sls_through_id        VARCHAR2(2),
active                     NUMBER(1)
)

当我运行代码时,我有错误这个

Caused by: java.sql.SQLException: ORA-06550: line 1, column 7:
PLS-00306: wrong number or types of arguments in call to 'PR_SP_FAHMI'
ORA-06550: line 1, column 7:
PL/SQL: Statement ignored

我在网上搜索过类似的错误来解决这个问题,但仍然无法解决我的问题。

要使用Spring调用一个Oracle存储过程,该过程接受一个类型为对象表的参数,需要做几件事。

首先,您需要创建一个对应于Oracle对象类型T_OBJECT_FROM_EXCEL的Java类。它需要实现SQLData接口。它看起来如下(请随意更改字段名称(:

import java.sql.SQLData;
import java.sql.SQLException;
import java.sql.SQLInput;
import java.sql.SQLOutput;
public class ExcelRow implements SQLData {
private String objectGroup;
private String objectCode;
private String prodMatrixId;
private String prodChannelId;
private String prodSalesGroupId;
private String prodCustGroup;
private String slsThroughId;
private boolean active;
// ... constructor, getters and setters omitted ...
@Override
public String getSQLTypeName() {
return "PARAMS.T_OBJECT_FROM_EXCEL";
}
@Override
public void readSQL(SQLInput sqlInput, String typeName) throws SQLException {
throw new SQLException("This has not been implemented");
}
@Override
public void writeSQL(SQLOutput sqlOutput) throws SQLException {
sqlOutput.writeString(this.objectGroup);
sqlOutput.writeString(this.objectCode);
sqlOutput.writeString(this.prodMatrixId);
sqlOutput.writeString(this.prodChannelId);
sqlOutput.writeString(this.prodSalesGroupId);
sqlOutput.writeString(this.prodCustGroup);
sqlOutput.writeString(this.slsThroughId);
sqlOutput.writeInt(this.active ? 1 : 0);
}
}

SQLData接口有三种方法,getSQLTypeName()readSQL()writeSQL()getSQLTypeName()返回我们要映射到的Oracle类型的名称,writeSQL()ExcelRow对象中的数据写入给定的输出对象sqlOutput。我们编写的字段的顺序必须与它们在T_OBJECT_FROM_EXCEL中声明的顺序相匹配。readSQL()用于将从数据库返回的T_OBJECT_FROM_EXCEL对象转换为ExcelRow,但我们只感兴趣的是向数据库发送值,而不是从数据库接收值,所以为了简洁起见,我只是实现了抛出异常。

其次,您需要调整调用存储过程的代码。您的方法executeSPForInsertData似乎有些通用,因为它采用了一个存储过程和一个任意的参数映射。由于表类型参数的原因,这个存储过程以通用方式调用有点困难,所以我们将专门为这个存储过程调用编写一个方法:

public void executeSPForInsertData(DataSource ds, ExcelRow[] excelRows){
String parameterName = "P_T_TABLE_UPLD_EXCEL";
SqlParameter[] parameterTypes = { new SqlParameter(parameterName, OracleTypes.ARRAY, "PARAMS.EXCEL") };
Map<String, Object> parameters =
Collections.singletonMap(parameterName, new SqlArrayValue<>(excelRows));
SimpleJdbcCall jdbcCall = new SimpleJdbcCall(ds)
.withSchemaName("PARAMS")
.withCatalogName("PKG_PARA_UPLD_VAL")
.withProcedureName("PR_SP_FAHMI")
.withoutProcedureColumnMetaDataAccess()
.declareParameters(parameterTypes);
jdbcCall.execute(parameters);
}

我们首先声明参数类型:有一个参数,它是一个数组,其Oracle类型在PARAMS模式中为EXCEL。然后我们声明参数值,使用SqlArrayValue来包装我们传入的ExcelRows的数组。最后我们设置存储过程调用,然后调用它。由于参数名称使用了两次,我将其放在一个局部变量中。

我对Oracle 18c XE数据库进行了快速测试,它成功了,因为我可以调用这个存储过程并让它将数据写入数据库。

最新更新