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

标签 java oracle jdbc plsql procedure

我有 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 接口(interface)具有三个方法: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 包装我们传入的 ExcelRow 数组。最后,我们设置存储过程调用并调用它。由于参数名称使用了两次,所以我将其放在局部变量中。

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

关于java - 由 : java. sql.SQLException : ORA-06550: line 1, 第 7 列引起:PLS-00306:调用 'PR_SP_FAHMI' 时参数数量或类型错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61423759/

相关文章:

java - 使用 PostgreSQL 数据库作为简单键值存储的最佳方式

java - android gradle构建错误?

java - getObject(index, Long) 对于 null 返回 0

jdbc - Jaybird 中的 Firebird NUMERIC/DECIMAL 精度和比例

oracle - 如何在oracle中插入日期和时间?

java - JPA 专业知识是否可以转移到 Toplink?

mysql - sql查询查找最流行的时间

java - 在具有非默认构造函数的类中如何初始化 @Autowired 对象?

java - Flyway 是否知道数据库状态或它是否专门验证迁移文件的不变性?

java - 自定义映射器类没有得到结果?