【问题标题】:How to convert a DBNULL to double?如何将 DBNULL 转换为双精度值?
【发布时间】:2011-01-21 23:04:53
【问题描述】:

我正在从数据库中获取一个值。它出现了一个错误,说“从 DBNULL 到 Double 的转换无效。”任何人,请帮助我?

公共函数 Total() 作为 Double Dim Total As Double

    Dim strConn As String
    strConn = ConfigurationManager.ConnectionStrings("***").ToString
    Dim conn As New SqlConnection(strConn)

    Dim strSql As String
    strSql = "SELECT SUM (p.Price * c.Quantity) as 'Total' " & _
    "FROM CartItem sci INNER JOIN Product p ON c.ProductID=p.ProductID " & _
    "WHERE c.CartID=@CartID "

    Dim cmd As New SqlCommand(strSql, conn)

    cmd.Parameters.AddWithValue("@CartID", CartID)

    Dim da As New SqlDataAdapter(cmd)

    Dim ds As New DataSet

    conn.Open()

    da.Fill(ds, "CartItem")

    conn.Close()

    If (ds.Tables("CartItem").Rows.Count) <> 0 Then
        **Total = ds.Tables("ShopCartItem").Rows(0)("Total")**
    Else
        Total = 0.0
    End If
    Return Total
End Function

【问题讨论】:

标签: asp.net visual-studio ado.net dbnull


【解决方案1】:

您不能将 DbNull 强制转换为 double。不过,您可以将其转换为可为空的 double(c# double?)。

我写了一堆 DataRow 扩展方法 (C#) 来帮助解决这个问题。使语法更加整洁。用法很简单。一个 C# 示例:

public class Hormone
{
    public int           ID               { get ; private set ; }
    public HormoneLuType Type             { get ; private set ; }
    public int           AgeStarted       { get ; private set ; }
    public int           AgeStopped       { get ; private set ; }
    public int           DurationInMonths { get ; private set ; }
    public bool          IsCurrentlyUsing { get ; private set ; }
    public DateTime?     DateLastEdited   { get ; private set ; }

    public string Name { get { return Type.ToString() } }

    public Hormone( DataRow dr )
    {
        this.ID               =                    dr.CastAsInt(              "ihormoneid" )        ;
        this.Type             = new HormoneLuType( dr.CastAsIntNullable(      "ihormluid"  ) ?? 0 ) ;
        this.AgeStarted       = (int)              dr.CastAsDecimal(          "nstartage"  )        ;
        this.AgeStopped       = (int)              dr.CastAsDecimal(          "nendage"    )        ;
        this.DurationInMonths = (int)              dr.CastAsDecimal(          "nduration"  )        ;
        this.IsCurrentlyUsing =                    dr.CastAsBool(             "lusingnow"  )        ;
        this.DateLastEdited   =                    dr.CastAsDateTimeNullable( "tedit"      )        ;

        return ;
    }
}

这是扩展类:

using System;
using System.Data;

namespace DataAccess.Utils
{
    public static class DataRowExtensions
    {

        #region downcast to DateTime

        public static DateTime CastAsDateTime( this DataRow row , int index )
        {
            return toDateTime( row[index] ) ;
        }
        public static DateTime CastAsDateTime( this DataRow row , string columnName )
        {
            return toDateTime( row[columnName] ) ;
        }

        public static DateTime? CastAsDateTimeNullable( this DataRow row , int index )
        {
            return toDateTimeNullable( row[index] );
        }
        public static DateTime? CastAsDateTimeNullable( this DataRow row , string columnName )
        {
            return toDateTimeNullable( row[columnName] ) ;
        }

        #region conversion helpers

        private static DateTime toDateTime( object o )
        {
            DateTime value = (DateTime)o;
            return value;
        }

        private static DateTime? toDateTimeNullable( object o )
        {
            bool  hasValue = !( o is DBNull );
            DateTime? value    = ( hasValue ? (DateTime?) o : (DateTime?) null ) ;
            return value;
        }

        #endregion

        #endregion downcast to DateTime

        #region downcast to byte[]

        public static byte[] CastAsByteArray( this DataRow row , int index )
        {
            return toByteArray( row[index] );
        }
        public static byte[] CastAsByteArray( this DataRow row , string columnName )
        {
            return toByteArray( row[columnName] );
        }

        #region conversion helpers

        private static byte[] toByteArray( object o )
        {
            bool   hasValue = !( o is DBNull );
            byte[] value    = ( hasValue ? (byte[]) o : (byte[]) null ) ;
            return value;
        }

        #endregion

        #endregion downcast to Byte[]

        #region downcast to int

        public static int CastAsInt( this DataRow row , int index )
        {
            return toInt( row[index] ) ;
        }
        public static int CastAsInt( this DataRow row , string columnName )
        {
            return toInt( row[columnName] ) ;
        }

        public static int? CastAsIntNullable( this DataRow row , int index )
        {
            return toIntNullable( row[index] );
        }
        public static int? CastAsIntNullable( this DataRow row , string columnName )
        {
            return toIntNullable( row[columnName] ) ;
        }

        #region conversion helpers

        private static int toInt( object o )
        {
            int value = (int)o;
            return value;
        }

        private static int? toIntNullable( object o )
        {
            bool hasValue = !( o is DBNull );
            int? value    = ( hasValue ? (int?) o : (int?) null ) ;
            return value;
        }

        #endregion

        #endregion downcast to int

        #region downcast to int

        public static decimal CastAsDecimal( this DataRow row , int index )
        {
            return toDecimal( row[index] ) ;
        }
        public static decimal CastAsDecimal( this DataRow row , string columnName )
        {
            return toDecimal( row[columnName] ) ;
        }

        public static decimal? CastAsDecimalNullable( this DataRow row , int index )
        {
            return toDecimalNullable( row[index] );
        }
        public static decimal? CastAsDecimalNullable( this DataRow row , string columnName )
        {
            return toDecimalNullable( row[columnName] ) ;
        }

        #region conversion helpers

        private static decimal toDecimal( object o )
        {
            decimal value = (decimal)o;
            return value;
        }

        private static decimal? toDecimalNullable( object o )
        {
            bool     hasValue = !( o is DBNull );
            decimal? value    = ( hasValue ? (decimal?) o : (decimal?) null ) ;
            return value;
        }

        #endregion

        #endregion downcast to int

        #region downcast to bool

        public static bool CastAsBool( this DataRow row , int index )
        {
            return toBool( row[index] ) ;
        }
        public static bool CastAsBool( this DataRow row , string columnName )
        {
            return toBool( row[columnName] ) ;
        }

        public static bool? CastAsBoolNullable( this DataRow row , int index )
        {
            return toBoolNullable( row[index] );
        }
        public static bool? CastAsBoolNullable( this DataRow row , string columnName )
        {
            return toBoolNullable( row[columnName] ) ;
        }

        #region conversion helpers

        private static bool toBool( object o )
        {
            bool value = (bool)o;
            return value;
        }

        private static bool? toBoolNullable( object o )
        {
            bool  hasValue = !( o is DBNull );
            bool? value    = ( hasValue ? (bool?) o : (bool?) null ) ;
            return value;
        }

        #endregion

        #endregion downcast to bool

        #region downcast to string

        public static string CastAsString( this DataRow row , int index )
        {
            return toString( row[index] );
        }
        public static string CastAsString( this DataRow row , string columnName )
        {
            return toString( row[columnName] );
        }

        #region conversion helpers

        private static string toString( object o )
        {
            bool   hasValue = !( o is DBNull );
            string value    = ( hasValue ? (string) o : (string) null ) ;
            return value;
        }

        #endregion

        #endregion downcast to string

    }
}

希望这会有所帮助!

【讨论】:

    【解决方案2】:

    尝试将您的 SQL 更改为:

    SELECT SUM (ISNULL(p.Price,0) * ISNULL(c.Quantity,0) ) as 'Total' " & _
        "FROM CartItem sci INNER JOIN Product p ON c.ProductID=p.ProductID " & _
        "WHERE c.CartID=@CartID 
    

    您还可以将数据库表设置为不允许空值

    【讨论】:

      【解决方案3】:

      您需要检查一下 ds.Tables("ShopCartItem").Rows(0))("Total") = DbNull.Value 在你设置之前。

      If (ds.Tables("CartItem").Rows.Count) <> 0 Then
          If ds.Tables("ShopCartItem").Rows(0))("Total") = DbNull.Value
             Total = 0.0
          else
            Total = ds.Tables("ShopCartItem").Rows(0)("Total")**
      Else
          Total = 0.0
      End If
      

      【讨论】:

      • 有效的方法!非常感谢!!!!!!!!!!!!我真的很感激。一百万谢谢你!!!! :)))))))))
      【解决方案4】:

      如果值可能为 null,您的返回类型需要为 double 吗? (可为空的双精度)。

      【讨论】:

      • 正是如此。 NULL 是缺失值。从概念上讲,它是一个位于该类型所有可能值域之外的值。修正反对票。
      【解决方案5】:

      您能否改为在 VB 端使用 Nullable(Of double) 作为您的数据类型?

      【讨论】:

        【解决方案6】:

        你必须检查Nullable(of Double).HasValue。 总计 = 0.0

        If (ds.Tables("CartItem").Rows.Count) <> 0 Then
             Dim sciValue As Nullable(Of Double) = DirectCast(ds.Tables("ShopCartItem").Rows(0)("Total"), Nullable(Of Double))
             If sciValue.HasValue Then
                 Total = sciValue.Value
             End If
        End If
        

        【讨论】:

          猜你喜欢
          • 2021-09-12
          • 2019-02-24
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 2012-12-04
          • 2020-08-09
          • 1970-01-01
          • 2011-01-14
          相关资源
          最近更新 更多