从 C# 调用 Oracle 存储过程?

Posted

技术标签:

【中文标题】从 C# 调用 Oracle 存储过程?【英文标题】:Calling Oracle stored procedure from C#? 【发布时间】:2011-04-25 19:56:42 【问题描述】:

如何从C#调用oracle中的存储过程?

【问题讨论】:

你能发布存储过程吗?您使用哪些库来连接数据库? ADO.NET? ORM(nHibernate,EF)?如果您想要一个适合您需求的答案,您需要提供更多详细信息。 如果您希望得到特定于 Oracle 的答案,请将您的问题标记为 Oracle。谢谢! 我以前使用过基本的 SQL 查询。但是,现在我想调用已经编写好的 Oracle 过程,使用 C# 代码。 【参考方案1】:

请访问这个由 oracle 为 Microsoft OracleClient 开发人员设置的 ODP 站点: http://www.oracle.com/technetwork/topics/dotnet/index-085703.html

下面还有一个示例代码,可以帮助您开始从 C# 调用存储过程到 Oracle。 PKG_COLLECTION.CSP_COLLECTION_HDR_SELECT 是基于 Oracle 构建的存储过程,接受参数 PUNIT、POFFICE、PRECEIPT_NBR 并在 T_CURSOR 中返回结果。

using Oracle.DataAccess;
using Oracle.DataAccess.Client;

public DataTable GetHeader_BySproc(string unit, string office, string receiptno)

    using (OracleConnection cn = new OracleConnection(DatabaseHelper.GetConnectionString()))
    
        OracleDataAdapter da = new OracleDataAdapter();
        OracleCommand cmd = new OracleCommand();
        cmd.Connection = cn;
        cmd.InitialLONGFetchSize = 1000;
        cmd.CommandText = DatabaseHelper.GetDBOwner() + "PKG_COLLECTION.CSP_COLLECTION_HDR_SELECT";
        cmd.CommandType = CommandType.StoredProcedure;
        cmd.Parameters.Add("PUNIT", OracleDbType.Char).Value = unit;
        cmd.Parameters.Add("POFFICE", OracleDbType.Char).Value = office;
        cmd.Parameters.Add("PRECEIPT_NBR", OracleDbType.Int32).Value = receiptno;
        cmd.Parameters.Add("T_CURSOR", OracleDbType.RefCursor).Direction = ParameterDirection.Output;

        da.SelectCommand = cmd;
        DataTable dt = new DataTable();
        da.Fill(dt);
        return dt;
    

【讨论】:

为什么首选InitialLONGFetchSize = 1000?来自文档Default = 0. Setting this property to 0 defers the LONG and LONG RAW data retrieval entirely until the application specifically requests it.【参考方案2】:

我现在已经掌握了从 C# 调用过程所需的步骤

   //GIVE PROCEDURE NAME
   cmd = new OracleCommand("PROCEDURE_NAME", con);
   cmd.CommandType = CommandType.StoredProcedure;

   //ASSIGN PARAMETERS TO BE PASSED
   cmd.Parameters.Add("PARAM1",OracleDbType.Varchar2).Value = VAL1;
   cmd.Parameters.Add("PARAM2",OracleDbType.Varchar2).Value = VAL2;

   //THIS PARAMETER MAY BE USED TO RETURN RESULT OF PROCEDURE CALL
   cmd.Parameters.Add("vSUCCESS", OracleDbType.Varchar2, 1);
   cmd.Parameters["vSUCCESS"].Direction = ParameterDirection.Output;

   //USE THIS PARAMETER CASE CURSOR IS RETURNED FROM PROCEDURE
   cmd.Parameters.Add("vCHASSIS_RESULT",OracleDbType.RefCursor,ParameterDirection.InputOutput); 

   //CALL PROCEDURE
   con.Open();
   OracleDataAdapter da = new OracleDataAdapter(cmd);
   cmd.ExecuteNonQuery();

   //RETURN VALUE
   if (cmd.Parameters["vSUCCESS"].Value.ToString().Equals("T"))
   
      //YOUR CODE
   
   //OR
   //IN CASE CURSOR IS TO BE USED, STORE IT IN DATATABLE
   con.Open();
   OracleDataAdapter da = new OracleDataAdapter(cmd);
   da.Fill(dt);

希望对你有帮助

【讨论】:

你能告诉我OracleDataAdapter da = new OracleDataAdatper(cmd);的目的是什么吗?需要这条线吗?【参考方案3】:

这与非查询命令的机制基本相同:

command.CommandText = 的名称 存储过程 command.CommandType = CommandType.StoredProcedure 对 command.Parameters.Add 的调用次数与 sp 所需的参数数量一样 command.ExecuteNonQuery

有很多例子,谷歌返回的第一个是this one

还有一个你可能会陷入的小陷阱,如果你的 SP 是一个函数,你的返回值参数必须在参数集合中排在第一位

【讨论】:

【参考方案4】:

这段代码很适合我调用 oracle 存储过程

通过在解决方案资源管理器中右键单击您的项目名称来添加引用>添加引用>.Net,然后添加命名空间。

using System.Data.OracleClient;
using System.Data;

然后将此代码粘贴到事件处理程序中

        string str = "User ID=username;Password=password;Data Source=Test";
        OracleConnection conn = new OracleConnection(str);
        OracleCommand cmd = new OracleCommand("stored_procedure_name", conn);
        cmd.CommandType = CommandType.StoredProcedure;
        --Ad parameter list--
        cmd.Parameters.Add("parameter_name", "varchar2").Value = value;
        ....
        conn.Open();
        cmd.ExecuteNonQuery();

它已经完成了......使用 C# 进行愉快的编码

【讨论】:

【参考方案5】:

连接到 Oracle 很糟糕。这是一些带有 using 语句的更简洁的代码。许多其他示例不会在他们创建的对象上调用 IDisposable 方法。

using (OracleConnection connection = new OracleConnection("ConnectionString"))
    using (OracleCommand command = new OracleCommand("ProcName", connection))             
    
          command.CommandType = CommandType.StoredProcedure;
          command.Parameters.Add("ParameterName", OracleDbType.Varchar2).Value = "Your Data Here";
          command.Parameters.Add("SomeOutVar", OracleDbType.Varchar2, 120);
          command.Parameters["return_out"].Direction = ParameterDirection.Output;
          command.Parameters.Add("SomeOutVar1", OracleDbType.Varchar2, 120);
          command.Parameters["return_out2"].Direction = ParameterDirection.Output;
          connection.Open();
          command.ExecuteNonQuery();
          string SomeOutVar = command.Parameters["SomeOutVar"].Value.ToString();
          string SomeOutVar1 = command.Parameters["SomeOutVar1"].Value.ToString();
    

【讨论】:

谢谢,这真的很整洁。 请注意,此示例使用 Oracle.ManagedDataAccess.Core nuget 包【参考方案6】:

在 .Net 至版本 4 中,这可以与 SQL Server 存储过程相同的方式完成,但请注意您需要:

using System.Data.OracleClient;

有some system requirements here,您应该验证在您的场景中是否正常。

Microsoft 是deprecating this namespace as of .Net 4,因此将来需要第三方提供商。考虑到这一点,您最好使用 go 一词中的Oracle Data Provider for .Net (ODP.NET) - 这具有 Microsoft 类中没有的优化。还有其他第三方选项,但甲骨文在保留 .Net 开发人员方面拥有强大的既得利益,因此他们应该是好的。

【讨论】:

【参考方案7】:

代替

cmd = new OracleCommand("ProcName", con);
cmd.CommandType = CommandType.StoredProcedure;
cmd.Parameters.Add("ParName", OracleDbType.Varchar2, ParameterDirection.Input).Value = "foo";

您也可以使用以下语法:

cmd = new OracleCommand("BEGIN ProcName(:p0); END;", con);
cmd.CommandType = CommandType.Text;
cmd.Parameters.Add("ParName", OracleDbType.Varchar2, ParameterDirection.Input).Value = "foo";

注意,如果您设置cmd.BindByName = False(这是默认值),那么您必须按照在命令字符串中写入的顺序添加参数,实际名称不相关。对于cmd.BindByName = True,参数名称必须匹配,顺序无关紧要。

如果是函数调用,命令字符串将如下所示:

cmd = new OracleCommand("BEGIN :ret := ProcName(:ParName); END;", con);
cmd.CommandType = CommandType.Text;
cmd.Parameters.Add("ret", OracleDbType.RefCursor, ParameterDirection.ReturnValue);    
cmd.Parameters.Add("ParName", OracleDbType.Varchar2, ParameterDirection.Input).Value = "foo";
// cmd.ExecuteNonQuery(); is not needed, otherwise the function is executed twice!
var da = new OracleDataAdapter(cmd);
da.Fill(dt);

【讨论】:

【参考方案8】:

以下在 .NET Core 解决方案中为我工作。请注意,它使用 OracleDataReader,Oracle CommandType 是 CommandType.Text

using Oracle.ManagedDataAccess.Client;

.......

                string spSql = "BEGIN STORED_PROC_NAME(:IN_PARAM, :OUT_PARAM1, :OUT_PARAM2); END; ";

                using (OracleConnection oraCnn = new OracleConnection(cnnString))
                using (OracleCommand oraCommand = new OracleCommand(spSql, oraCnn))
                
                    await oraCnn.OpenAsync(cancellationToken);
                    oraCommand.CommandType = CommandType.Text;
                    oraCommand.BindByName = true;

                    oraCommand.Parameters.Add("IN_PARAM", OracleDbType.Long, ParameterDirection.Input).Value = 123;
                    oraCommand.Parameters.Add("OUT_PARAM1", OracleDbType.Int32, null, ParameterDirection.Output);
                    oraCommand.Parameters.Add("OUT_PARAM2", OracleDbType.Varchar2, 4000, null, ParameterDirection.Output);

                    OracleDataReader objReader = oraCommand.ExecuteReader();

                    string outParamValue= oraCommand.Parameters["OUT_PARAM2"].Value.ToString();
                

【讨论】:

以上是关于从 C# 调用 Oracle 存储过程?的主要内容,如果未能解决你的问题,请参考以下文章

c# 调用Oracle存储过程 PLS-00201:必须声明标识符

c#调用oracle存储过程,返回2个值

在c#中调用oracle存储过程

执行 C#的服务器端程序调用Oracle的存储过程报错

C# 如何调用此存储过程

[C#对Oracle操作]C#操作调用Orcale存储过程有参数