我正在尝试从C#调用oracle存储过程'spValidateDBA',但我总是收到错误 - “PLS-00306:调用'SPVALIDATEDBA'时参数的数量或类型错误”,同时尝试这样做
尝试从C#调用存储过程参数'UserRole'时可能会出现问题,它的类型在C#代码中设置不正确,我不知道如何正确设置它。有人请帮帮我在这里,我真的需要尽快解决这个问题。
C#代码: -
string a = txtUsrId.Text.ToUpper();
string sConnectionString = "Data Source=XE;User ID=sys;Password=system;DBA
PRIVILEGE=sysdba";
OracleConnection myConnection = new OracleConnection(sConnectionString);
OracleCommand myCommand = new OracleCommand("spValidateDBA", myConnection);
myCommand.CommandType = CommandType.StoredProcedure;
myCommand.CommandText = "spValidateDBA";
myCommand.Parameters.Add("UserId", OracleDbType.Varchar2, 50);
myCommand.Parameters["UserId"].Value = a;
myCommand.Parameters.Add("UserRole",OracleDbType.Varchar2, 50).Direction =
ParameterDirection.Output;
myCommand.Parameters.Add("UserIdOut", OracleDbType.Varchar2, 50).Direction =
ParameterDirection.Output;
var rolechk = false;
string checkrole = "DBA";
myConnection.Open();
myCommand.ExecuteReader();
//this is where i'm getting the error "PLS-00306: wrong number
//or types of arguments in call to 'SPVALIDATEDBA'"
存储过程spValidateDBA
create or replace PROCEDURE spValidateDBA(
UserId IN VARCHAR2,
UserRole OUT STRING_ARRAY,
UserIdOut OUT VARCHAR2)
AS
BEGIN
EXECUTE IMMEDIATE 'select GRANTED_ROLE bulk collect into UserRole from DBA_USERS DU,DBA_ROLE_PRIVS DRP where DU.USERNAME=UserId AND DU.USERNAME=DRP.GRANTEE';
EXECUTE IMMEDIATE 'select USERNAME into UserIdOut from DBA_USERS DU where DU.USERNAME=UserId';
END spValidateDBA;
STRING_ARRAY是在此语句中声明的字符串数组类型:
CREATE TYPE STRING_ARRAY AS TABLE OF VARCHAR2(50);
如有任何问题,请与我们联系。
答案 0 :(得分:0)
命令文本必须如下:
myCommand.CommandText = "spValidateDBA(:UserId, :UserRole, :UserIdOut )";
然后你必须调用myCommand.ExecuteNonQuery();
而不是ExecuteReader。
myCommand.Parameters.Add("UserRole",OracleDbType.Varchar2, 50)
也是错误的。我认为它应该类似于这个:
myCommand.Parameters.Add("UserRole");
myCommand.Parameters["UserRole"].CollectionType = OracleCollectionType.PLSQLAssociativeArray;
myCommand.Parameters["UserRole"].Direction = ParameterDirection.Output;
我从未使用PLSQLAssociativeArray
作为输出参数,所以我不确定细节。也许你的程序可能会返回一个SYS_REFCURSOR。
在PL / SQL过程中,我认为不需要动态SQL,简单起见:
create or replace PROCEDURE spValidateDBA(
UserId IN VARCHAR2, UserRole OUT STRING_ARRAY, UserIdOut OUT VARCHAR2)
AS
BEGIN
select GRANTED_ROLE bulk collect into UserRole from DBA_USERS DU,DBA_ROLE_PRIVS DRP where DU.USERNAME=UserId AND DU.USERNAME=DRP.GRANTEE;
select USERNAME into UserIdOut from DBA_USERS DU where DU.USERNAME= UserId';
END spValidateDBA;