簡體   English   中英

如何從帶有 SPROC 參數的 SubSonic SPROC 調用中返回強類型 object?

[英]How to return a strongly-typed object from a SubSonic SPROC call with SPROC parameters?

我有一個返回 ItemDatumCollection 類型的 SubSonic 集合的數據調用方方法。

存儲過程執行如下:

itemDatumCollection.LoadAndCloseReader(sp.GetReader());

但是,我無法以這種方式訪問 sproc 的 output 參數,因為我可以在調用 sp.GetDataSet() 時執行以下操作:

            itemsDataSet = sp.GetDataSet();

            actualNumberOfResults = ((Int64)sp.OutputValues[1]);
            numberOfResultsReturned = ((Int64)sp.OutputValues[2]);

有沒有辦法使用第一種方法訪問 sproc 的 output 參數 - 從我的 sproc 調用返回一個 SubSonic 強類型集合?

謝謝。

我想做類似的事情,最后改變了大部分 StoredProcedures.tt 文件,我認為是 SubSonic 的一些內部結構。 在沒有太詳細的情況下,我更改了 by.tt 文件,以便它會為給定的存儲過程生成此文件:

    public void COMPANIES_ACTIVATE_PACKAGE(long PI_COMPANY_ID, string PI_ACTIVE, long PI_USER_ID, out long PO_ERRCODE, out string PO_ERRMSG, out string PO_ORA_ERRMSG){
        StoredProcedure sp=new StoredProcedure("COMPANIES.ACTIVATE_PACKAGE",this.Provider);
        sp.Command.AddParameter("PI_COMPANY_ID",PI_COMPANY_ID,DbType.Decimal);
        sp.Command.AddParameter("PI_ACTIVE",PI_ACTIVE,DbType.AnsiString);
        sp.Command.AddParameter("PI_USER_ID",PI_USER_ID,DbType.Decimal);
        sp.Command.AddOutputParameter("PO_ERRCODE",DbType.AnsiString);
        sp.Command.AddOutputParameter("PO_ERRMSG",DbType.AnsiString);
        sp.Command.AddOutputParameter("PO_ORA_ERRMSG",DbType.AnsiString);
        sp.Execute();
        var prms = sp.Command.Parameters;
        PO_ERRCODE = ConvertValue<long>(prms.GetParameter("PO_ERRCODE").ParameterValue);
        PO_ERRMSG = ConvertValue<string>(prms.GetParameter("PO_ERRMSG").ParameterValue);
        PO_ORA_ERRMSG = ConvertValue<string>(prms.GetParameter("PO_ORA_ERRMSG").ParameterValue);
    }

所以基本上我傳入任何 sproc 參數,並為返回值定義了“out”參數。

這里也沒有顯示,但如果我有一個 InOut 參數,那么我通過ref而不是out傳遞它。

然后在我的實際應用程序代碼中,我可以像任何其他 function 一樣調用存儲的過程:

long errorCode;
string errorMsg, oraErrorMsg;

db.COMPANIES_ACTIVATE_PACKAGE(123, "Y", 456, out errorCode, out errorMsg, out oraErrorMsg);

if(errorCode > 0)
    // ... handle error...

我不知道這是否會直接插入 SS 而無需進一步更改,但這是 my.tt 文件。 您也許可以使用它,或者至少了解 go 的位置:

存儲過程.tt

<#@ template language="C#" debug="False" hostspecific="True"  #>
<#@ output extension=".cs" #>
<#@ include file="DB2DataProvider.ttinclude" #>
<#
    var sps = GetSPs(); 
    if(sps.Count>0){ 
#>  
using System;
using System.Data;
using System.ComponentModel;
using SubSonic;
using SubSonic.Schema;
using SubSonic.DataProviders;

namespace <#=Namespace#>{
    public partial class <#=DatabaseName#>DB{

        public T ConvertValue<T>(object paramVal)
        {
            if (paramVal == null || Convert.IsDBNull(paramVal))  // if the value is null, return the default for the desired type.
                return default(T);
            if (typeof(T) == paramVal.GetType())  // if types are already equal, no conversion needed. just cast.
                return (T)paramVal;
            else // types don't match. try to convert.
            {
                var conversionType = typeof(T);
                if (conversionType.IsGenericType && conversionType.GetGenericTypeDefinition().Equals(typeof(Nullable<>)))
                {
                    NullableConverter nullableConverter = new NullableConverter(conversionType);
                    conversionType = nullableConverter.UnderlyingType;
                }
                return (T)Convert.ChangeType(paramVal, conversionType);
            }
        }

<#  foreach(var sp in sps){#>
        public void <#=sp.CleanName#>(<#=sp.ArgList#>){
            StoredProcedure sp=new StoredProcedure("<#=sp.Name#>",this.Provider);
<#      foreach(var par in sp.Parameters) {
          if(par.In && !par.Out) {#>
            sp.Command.AddParameter("<#=par.Name#>",<#=par.CleanName#>,DbType.<#=par.DbType#>);
<#        } else if(!par.In && par.Out) {#>
            sp.Command.AddOutputParameter("<#=par.Name#>",DbType.<#=par.DbType#>);
<#        } else {#>
            sp.Command.AddParameter("<#=par.Name#>",<#=par.CleanName#>,DbType.<#=par.DbType#>,ParameterDirection.InputOutput);
<#      }}#>
            sp.Execute();
<#      bool hasOut = false;
        foreach(var par in sp.Parameters) {
          if(par.Out)
            hasOut = true;
        }
        if(hasOut) {
#>
            var prms = sp.Command.Parameters;
<#      }
        foreach(var par in sp.Parameters) {
          if(par.Out) {#>
            <#=par.Name#> = ConvertValue<<#=par.SysType#>>(prms.GetParameter("<#=par.Name#>").ParameterValue);
<#      }}#>
        }
<#  }#>

    }

}
<#  }#>

我很確定我還必須更改在數據庫 provider.ttinclude 文件中加載存儲的 proc 數據的查詢,以加載指示它是 In、Out 還是 InOut 參數類型的列。

...希望這在某種程度上有所幫助...

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM