1

使用 SqlDataAdapter 执行存储过程并返回结果集时遇到一个奇怪的问题。该过程返回单个字符串值,并且在 SSMS 中执行时正确执行并返回正确的值。但由于某种原因,DataSet 的 SqlDataAdapter.Fill 导致结果用前导 0 字符填充。

例如,在数据库中执行 sp 返回“175190336289169307”,但通过 .Fill 方法(或 ExecuteScalar)执行返回“00000000000000175190336289169307”。

有什么想法吗?有什么方法可以转换或转换此结果以使其正确吗?谢谢。

Dim param As SPParam
Dim r As String = ""

_cn = New SqlConnection(_connection)
_cn.Open()

_da = New SqlDataAdapter
_cmd = _cn.CreateCommand

_cmd.CommandType = CommandType.StoredProcedure
_cmd.CommandText = _StoredProcName


For Each param In _SPParams
    _cmd.Parameters.Add(param.SQLParam)
Next

Select Case Type
    Case ExecuteType.Scalar
        r = _cmd.ExecuteScalar.ToString
    Case ExecuteType.CommandOnly
        r = _cmd.ExecuteNonQuery
    Case ExecuteType.Tabular
        _da.SelectCommand = _cmd
        _ds = New DataSet
        _da.Fill(_ds)
End Select

For Each param In _SPParams
    param.Value = param.SQLParam.Value
Next

Return r

这是 SQL 代码的骨架。真的很简单。有一个调用底层标量函数的包装器 sp。函数返回值类型(最终也是 sp 的返回类型)是 VARCHAR(255)。返回值以空字符串开始,并一次构建一个块以创建返回字符串。在 SSMS 中调用函数或包装器 sp 仅返回最终字符串。没有前导 0 字符。

-- wrapper stored procedure
CREATE PROCedure [dbo].[sp_fubar] 
    @p_1 VARCHAR(120),
    @p_Status INT = 0 Output ,
    @p_ErrMsg VARCHAR(1000) = '' Output
AS

BEGIN TRY
    Set NOCOUNT ON;

    Set @p_Status = 99
    Set @p_ErrMsg = 'Unknown error'

    SELECT dbo.BaseFunction(@p_1) as [Result]

    Set @p_Status = 0
    Set @p_ErrMsg = 'Success'


END TRY
BEGIN CATCH
    Set @p_Status = 16
    Set @p_ErrMsg = 'Error in sp_fubar: ' + ERROR_MESSAGE()
END CATCH;

GO

-- base function call
CREATE FUNCTION [dbo].[BaseFunction] (
    @p_InputData VARCHAR(40)
)
RETURNS VARCHAR(255)
AS
BEGIN
    -- Declare the return variable here
    DECLARE @p_ReturnData VARCHAR(255)

    declare
        @len as integer,
        @c as integer

    -- Trim off access spaces
    set @p_InputData = ltrim(rtrim(@p_InputData))

    -- Get the length of the given data
    set @len = len(@p_InputData)

    while (@len >= 1)
    begin
        -- Get the ascii value of each character
        set @c = ascii(substring(@p_InputData, @len, 1))

        -- Do stuff with @c

        set @p_ReturnData = @p_ReturnData + cast(@c as varchar)
        set @len = @len - 1 
    end

    -- Return the result of the function
    RETURN @p_ReturnData

END

GO
4

1 回答 1

0

原来是垃圾中垃圾的情况。

从 .net 应用程序传递给 sp 的 varchar @p_1 参数来自 String 类型变量。该变量被定义为长度为 20,因为这是从中读取该变量的源的定义长度。但在实践中,所需字符串的长度(就非空白字符和非空字符而言)小于 20。但完整的字符串、空字符和所有字符都被传递给 sp,这就是填充。

没有找到直接指定或重新标注.net中字符串类型长度的方法,我添加了一个快速循环来修剪字符串,问题解决了。

不过很难找到,因为在 Visual Studio 中查看字符串的值会看到所需的字符串(不显示非打印字符),因此实际字符串更长并不立即明显。直到我查看 .Len 属性并看到值为 20 时,它才单击。

于 2013-10-05T12:13:21.927 回答