[使用ADO vb6的带有参数的VBA调用存储过程

问题描述 投票:1回答:2

我正在尝试使用存储过程将VB6的值插入SQL Server,但是当想要执行该过程时,它向我发送了以下错误消息

精度无效

我已经改变了很多次,试图输入它们的方式使我不再知道该怎么做。

这是VB6中的代码

Dim strSQL As String
strSQL = "INSERT INTO FacturasPendientes (IdFactura,FechaFactura,CodigoProveedor,NombreProveedor," _
       & " SubTotal,Iva,Total,FechaVencimiento,DiasDescuento,DescProntoPago,Pagado) VALUES (?,?,?,?,?,?,?,?)"

Dim CmdCont As ADODB.Command
Set CmdCont = New ADODB.Command

With CmdCont
    Set .ActiveConnection = Cnn
    .CommandType = adCmdStoredProc
    .CommandText = "sp_FacturasCompras"
    .Parameters.Append CmdCont.CreateParameter("@IdFactura", adVarChar, adParamInput, 50, TxtFactura.Text)
    .Parameters.Append CmdCont.CreateParameter("@FechaFactura", adDate, adParamInput, 50, dtpFactura.Value)
    .Parameters.Append CmdCont.CreateParameter("@CodigoProveedor", adVarChar, adParamInput, 50, TxtCodigoProveedor.Text)
    .Parameters.Append CmdCont.CreateParameter("@NombreProveedor", adVarChar, adParamInput, 100, txtProveedor.Text)
    .Parameters.Append CmdCont.CreateParameter("@SubTotal", adNumeric, adParamInput, 18, Round(Val(m_Subtotal), 2))
    .Parameters.Append CmdCont.CreateParameter("@Iva", adNumeric, adParamInput, 18, Round(Val(m_Iva), 2))
    .Parameters.Append CmdCont.CreateParameter("@Total", adNumeric, adParamInput, 18, Round(Val(m_Total), 2))
    .Parameters.Append CmdCont.CreateParameter("@FechaVencimiento", adDate, adParamInput, 50, FechaFinal)
    .Parameters.Append CmdCont.CreateParameter("@DiasDescuento", adVarChar, adParamInput, 50, txtDias.Text)
    .Parameters.Append CmdCont.CreateParameter("@DescProntoPago", adVarChar, adParamInput, 50, txtDescuento.Text)
    .Parameters.Append CmdCont.CreateParameter("@Pagado", adBigInt, adParamInput, chkPago.Value)
    .Prepared = True
    .Execute
End With
MsgBox "Se Grabaron los datos"

并且在存储过程中是这个

ALTER PROCEDURE [dbo].[sp_FacturasCompras] 
    -- agregamos los valores de la consulta
    @IdFactura varchar(50),
    @FechaFactura date,
    @CodigoProveedor varchar(50),
    @NombreProveedor varchar(100),
    @SubTotal numeric(18),
    @Iva numeric(18),
    @Total numeric(18),
    @FechaVencimiento date,
    @DiasDescuento varchar(50),
    @DescProntoPago varchar(50),
    @Pagado bigint
AS
BEGIN
    INSERT INTO FacturasPendientes (IdFactura,FechaFactura,CodigoProveedor,NombreProveedor,
            SubTotal,Iva,Total,FechaVencimiento,DiasDescuento,DescProntoPago,Pagado)
    VALUES (@IdFactura,@FechaFactura,@CodigoProveedor,
            @NombreProveedor,@SubTotal,@Iva,@Total,@FechaVencimiento,@DiasDescuento,@DescProntoPago,@Pagado)
END
sql-server vb6 adodb
2个回答
0
投票

最简单的调用存储过程的方法是使用Cnn连接对象,像这样

Cnn.sp_FacturasCompras TxtFactura.Text, dtpFactura.Value, _
    TxtCodigoProveedor.Text, txtProveedor.Text, _
    Round(Val(m_Subtotal), 2), Round(Val(m_Iva), 2), _
    Round(Val(m_Total), 2), FechaFinal, txtDias.Text, _
    txtDescuento.Text, chkPago.Value

。 。 。看起来像Cnn对象上的普通方法调用。

如果您have使用ADODB.Command,那么最好是使用一些帮助程序函数来初始化ADODB.Parameter实例和ADODB.Command实例,像这样

Option Explicit

Private m_Subtotal      As Double
Private m_Iva           As Double
Private m_Total         As Double

Private Sub Form_Load()
    Dim oCmd As ADODB.Command

    Set oCmd = InitCommandHelper("sp_FacturasCompras", _
        InitParamHelper(TxtFactura.Text, adVarChar, 50), _
        InitParamHelper(dtpFactura.Value, adDBDate), _
        InitParamHelper(TxtCodigoProveedor.Text, adVarChar, 50), _
        InitParamHelper(txtProveedor.Text, adVarChar, 100), _
        InitParamHelper(Round(Val(m_Subtotal), 2), adDecimal, Precision:=18, NumericScale:=2), _
        InitParamHelper(Round(Val(m_Iva), 2), adDecimal, Precision:=18, NumericScale:=2), _
        InitParamHelper(Round(Val(m_Total), 2), adDecimal, Precision:=18, NumericScale:=2), _
        InitParamHelper(FechaFinal, adVarChar, 50), _
        InitParamHelper(txtDias.Text, adVarChar, 50), _
        InitParamHelper(txtDescuento.Text, adVarChar, 50), _
        InitParamHelper(chkPago.Value, adBigInt))
    Debug.Print oCmd.Parameters.Count
End Sub

Public Function InitCommandHelper( _
            ByVal sStoredProc As String, _
            ParamArray Params() As Variant) As ADODB.Command
    Const FUNC_NAME     As String = "InitCommandHelper"
    Dim lIdx            As Long

    On Error GoTo EH
    Set InitCommandHelper = New ADODB.Command
    With InitCommandHelper
        If InStr(sStoredProc, ".") > 0 Then
            .CommandText = sStoredProc
        Else
            .CommandText = "dbo." & sStoredProc
        End If
        .CommandType = adCmdStoredProc
        .Parameters.Append .CreateParameter("RetVal", adInteger, adParamReturnValue)
        lIdx = LBound(Params)
        Do While lIdx <= UBound(Params)
            .Parameters.Append Params(lIdx)
            lIdx = lIdx + 1
        Loop
    End With
    Exit Function
EH:
    Debug.Print "Critical error: " & Err.Description & "[Form1." & FUNC_NAME & "]", Timer
End Function

Public Function InitParamHelper( _
            ByVal vValue As Variant, _
            ByVal eType As ADODB.DataTypeEnum, _
            Optional ByVal Size As Long, _
            Optional Direction As ADODB.ParameterDirectionEnum = adParamInput, _
            Optional ByVal Name As String, _
            Optional ByVal Precision As Long, _
            Optional ByVal NumericScale As Long) As ADODB.Parameter
    Const FUNC_NAME     As String = "InitParamHelper"
    Const DBL_NUM_LIMIT As Double = 10# ^ 12 ' 922337203685477#
    Const EMPTY_GUID    As String = "{00000000-0000-0000-0000-000000000000}"

    On Error GoTo EH
    Select Case eType
    Case adVarChar, adVarWChar, adChar, adWChar
        '--- trim varchar params'
        If Not IsNull(vValue) And Not IsEmpty(vValue) Then
            vValue = Left$(vValue, Size)
        End If
    Case adLongVarChar, adLongVarWChar, adLongVarBinary
        '--- fix default size for BLOB params'
        If Size = 0 Then
            Size = -1
        End If
    Case adDBTimeStamp
        If CDate(vValue) = CLng(CDate(vValue)) Then
            eType = adDBDate
        ElseIf CLng(CDate(vValue)) = 0 Then
            eType = adDBTime
        Else
            eType = adDate
        End If
    Case adNumeric, adDecimal, adCurrency
        '--- numeric range'
        If IsNumeric(vValue) Then
            If vValue > DBL_NUM_LIMIT Then
                vValue = DBL_NUM_LIMIT
            ElseIf vValue < -DBL_NUM_LIMIT Then
                vValue = -DBL_NUM_LIMIT
            End If
        End If
    Case adGUID
        If Direction = adParamInput Then
            If IsEmpty(vValue) Then
                vValue = EMPTY_GUID
            End If
        End If
    End Select
    Set InitParamHelper = New ADODB.Parameter
    InitParamHelper.Name = Name
    InitParamHelper.Type = eType
    InitParamHelper.Direction = Direction
    InitParamHelper.Size = Size
    InitParamHelper.Value = vValue
    If Precision > 0 Then
        InitParamHelper.Precision = Precision
    End If
    If NumericScale > 0 Then
        InitParamHelper.NumericScale = NumericScale
    End If
    Exit Function
EH:
    Debug.Print "Critical error: " & Err.Description & "[Form1." & FUNC_NAME & "]", Timer
    Set InitParamHelper = Nothing
End Function

请注意,NUMERIC(18)被解析为NUMERIC(18, 0),它在浮点数后有0位数字,但是您对这些参数使用Round(..., 2),因此很可能是错误。只需使用NUMERIC(18, 2)或更好的Evern DECIMAL(19, 2),因为精度18和精度19使用相同的5个字节进行存储。

还要注意,sp_前缀是为系统存储过程保留的,请勿将其用于用户定义的存储过程。您必须为usp_提出不同的前缀,例如sp(不带下划线)。


0
投票

我怀疑问题出在adNumeric列。尝试如下代码:

Dim p as Parameter

With CmdCont
   Set p = .CreateParameter("@SubTotal", adNumeric, adParamInput)
   p.Precision = 18
   p.NumericScale = 2
   p.Value = Round(Val(m_Subtotal), 2)
   .Parameters.Append p
End With

您可以根据需要调整精度和比例。

© www.soinside.com 2019 - 2024. All rights reserved.