1 问题重现

(1)新建项目DBNullExp。项目属性为“控制台应用程序”;

(2)在项目下新建数据集Schools(数据集文件的后缀名为.xsd);

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQveWwyaXNvZnQ=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="">

(3)在数据集下新建数据表Students,表字段的定义例如以下表所看到的:

字段名

说明

ID

dc.DataType = Type.GetType("System.Int32");//类型

dc.AutoIncrement = true;//自己主动增量

dc.AutoIncrementSeed = 1;//起始为1

dc.AutoIncrementStep = 1;//步长为1

dc.AllowDBNull = false;//非空

Name

dc.DataType = Type.GetType("System.String");//类型

dc.AllowDBNull = false;//非空

Age

dc.DataType = Type.GetType("System.Int32");//类型

dc.AllowDBNull = true;//可空

Email

dc.DataType = Type.GetType("System.String");//类型

dc.AllowDBNull = true;//可空

(4)向Students数据表插入5条演示样例数据。

Schools.StudentsDataTable sdt = new Schools.StudentsDataTable();
sdt.Rows.Add(new object[] { null, "张三", 25, "zhangsan@sina.com" });
sdt.Rows.Add(new object[] { null, "李四", 23, "lisi@sina.com" });
sdt.Rows.Add(new object[] { null, "王五", 24, "wangwu@sina.com" });
sdt.Rows.Add(new object[] { null, "周六", null, null });
sdt.Rows.Add(new object[] { null, "吴七", 25, "wuqi@sina.com" });

(5)查询数据表sdt,获取当中年龄大于24的学生信息输出到界面。

var newStudentList = sdt.Where(it => it.Age > 25).ToList();
foreach (var s in newStudentList)
{
Console.WriteLine(s.Name);
}

(6)执行代码,抛出例如以下图所看到的异常。

2 问题分析与解决

从异常提示信息能够看出,问题出在Students表的Age字段上,异常信息中指出Age字段的值为DBNull,那么什么是DBNull。它与我们常常见到的Null又有什么差别?

Null指的是无效的对象引用。而DBNull是一个类。DBNull.Value是它唯一的实例。DBNull的实例DBNull.Value是数据库表中的空数据在.Net代码中的表现形式。我们知道,当数据库表中的可空字段没有被赋值时,数据库表中的该字段会被指定为Null,那么这个Null值在.Net代码中该怎样表示呢,使用的就是DBNull.Value。反过来说。在代码中。当一个字段的值等于DBNull.Value,那就说明这个字段在数据库中保存的值为Null。也就是说它在数据库中的值为空。所以,DBNull.Value对象指向有效的对象。并不像Null没有指向不论什么有效的对象。

再来看看我们实例中Students表的数据。例如以下图所看到的。

从图中能够看出。名为“周六”的学生的Age字段的值为空,确实存在异常信息所说的Age值为DBNull的情形。

那么,为什么,Age字段的值为DBNull就会抛出异常呢?我不就是想获取一下学生的Age值吗,没有值返回给我一个DBNull.Value不就得了吗。为什么会抛出异常呢?

继续深挖异常源头。通过查看异常的InnerException信息,得知该异常的内部异常为“指定的转换无效”,并定位到异常产生的问题代码。

[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Data.Design.TypedDataSetGenerator", "4.0.0.0")]
public int Age {
get {
try {
return ((int)(this[this.tableStudents.AgeColumn]));
}
catch (global::System.InvalidCastException e) {
throw new global::System.Data.StrongTypingException("表“Students”中列“Age”的值为DBNull。 ", e);
}
}
set {
this[this.tableStudents.AgeColumn] = value;
}
}

问题代码为:

return ((int)(this[this.tableStudents.AgeColumn]));

查看问题代码中this[this.tableStudents.AgeColumn]的值。为DBNull.Value。再将这个DBNull.Value隐式转换成整型数据时产生了异常。由于两者之间根本无法进行类型转换。这段代码是在新建数据集及创建表时自己主动生成的。所以改动需谨慎。

可是不知道当时微软为什么不在这里加上一个特殊处理,要是返回值为DBNull.Value。就直接返回0。为什么不这样做。就不去瞎推測了。

找到解决方法要紧。

3 解决方法

我们除了使用“it.Age”的方式来获取字段的值外,还能够使用“it["Age"]”的方式来获取表中的字段值。且知道当数据库字段值为空时,“it["Age"]”获取到的值会是DBNull.Value且不会抛出异常。所以我们就能够使用“it["Age"]”方式得到Age的值再与DBNull.Value比較。当不相等时再进行类型转换操作。这样一切就OK啦。改动后的代码例如以下所看到的。

var newStudentList1 = sdt.Where(it => it["Age"] != DBNull.Value && (int)it["Age"] > 24).ToList();

再次运行程序,得到以下想要的结果。

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQveWwyaXNvZnQ=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="">

因DataTable的字段值为DBNull引发的异常的更多相关文章

  1. C#三种判断数据库中取出的字段值是否为空(NULL) 的方法

    操作数据库,需要判断返回的字段值是否为空,收集了3种方法供参考 1 通过System.DBNull判断,网上大部分都使用这个方法. DataTable dt;                     ...

  2. 程序处理数据库中值字段值为null的查询显示

    1.如果你做了一个简单的注册界面,需要用户进行注册,但有些项是不必要填的,当用户完成注册时,数据库表中的相应字段的值会写入null,但如何将查询的字段的值null显示出来? 2.首先我们学习一下如何向 ...

  3. 使用泛型集合取代datatable作为返回值实现面向对象

    开会的时候,师父说.我们在机房重构时,尽量不要用datatable作为返回值.改用泛型集合的方式,这样能够实现真正的面向对象. 通过查资料和同学交流,把这个问题给攻克了. 对于泛型集合.我也有了一些认 ...

  4. datagridcolumn单元格怎么显示查询到的某个表的字段值(字段值可能为多个)

    例如,在之前做的项目中,查询mhz_xckcr表,select出某个业务的现场勘察人信息,select出的现场勘察人姓名(可能有多个)要在前台datagrid的一个datagridcolmn单元格显示 ...

  5. Mysql按照字段值做分组行转列查询

    今天做个后台服务,有个需求是批量生成一批表的数据,如果用BulkInsert会提升很大一截提交效率,但是如果用循环构造提交的Datable,则算法开销太高,所以用这种查询批量查出符合格式的DataTa ...

  6. 数据库表设计时一对一关系存在的必要性 数据库一对一、一对多、多对多设计 面试逻辑题3.31 sql server 查询某个表被哪些存储过程调用 DataTable根据字段去重 .Net Core Cors中间件解析 分析MySQL中哪些情况下数据库索引会失效

    数据库表设计时一对一关系存在的必要性 2017年07月24日 10:01:07 阅读数:694 在表设计过程中,我无意中觉得一对一关系觉得好没道理,直接放到一张表中不就可以了吗?真是说,网上信息什么都 ...

  7. SQL Server 动态行转列(参数化表名、分组列、行转列字段、字段值)

    一.本文所涉及的内容(Contents) 本文所涉及的内容(Contents) 背景(Contexts) 实现代码(SQL Codes) 方法一:使用拼接SQL,静态列字段: 方法二:使用拼接SQL, ...

  8. MSCRM CRM 获取PickList 字段值函数解决方案

    表单中有很多picklist字段 不想写链接stringmap代码: 实体ID查询方法: SELECT ObjectTypeCode from Entity  where  name='实体名称' 调 ...

  9. Mysql Sql语句令某字段值等于原值加上一个字符串

    MySQL连贯字符串不能利用加号(+),而利用concat. 比方在aa表的name字段前加字符'x',利用: update aa set name=concat('x',name); 替换: UPD ...

随机推荐

  1. [BZOJ 1150] 数据备份

    Link:https://www.lydsy.com/JudgeOnline/problem.php?id=1150 Solution: 思路和洛谷P1484完全相同 只不过将求最大不相邻的点权改为最 ...

  2. [CF418E]Tricky Password

    题意:有一个无限行$n$列的数表$a_{i,j}$,对于第$i\geq2$行,$a_{i,j}$为$a_{i-1,j}$在$a_{i-1,1\cdots j}$中出现的次数,要维护这个数表,支持修改第 ...

  3. 【循环节】 Codeforces Round #401 (Div. 2) A. Shell Game

    容易发现存在循环节. #include<cstdio> using namespace std; int n,x,a[3][6]={{0,1,2,2,1,0},{1,0,0,1,2,2}, ...

  4. 【计算几何】URAL - 2101 - Knight's Shield

    Little Peter Ivanov likes to play knights. Or musketeers. Or samurai. It depends on his mood. For pa ...

  5. 【博弈论】【SG函数】bzoj1457 棋盘游戏

    一开始就必胜的特判一下. #include<cstdio> #include<cstring> #include<set> #include<algorith ...

  6. 【bzoj1486】【[HNOI2009]梦幻布丁】启发式链表合并(详解)

    (画师当然是武内崇啦) Description N个布丁摆成一行,进行M次操作.每次将某个颜色的布丁全部变成另一种颜色的,然后再询问当前一共有多少段颜色.例如颜色分别为1,2,2,1的四个布丁一共有3 ...

  7. BUG YII2.0 cURL error 6: Could not resolve host:

    BUG描述:登录直接显示 原因:服务器设置端口权限,或者DNS毛病 解决方案:只能去服务器端设置,配置端口 DNS: 修改dns 114.114.114.114 或者 8.8.8.8

  8. identifier is too long 异常处理

    修改了oracle中的表. 报 identifier is too long 错误 我执行的脚本是: ---备份create table MDT_AGREEMENTMANAGEMENT_2018080 ...

  9. Vmware+Virtualbox+Ubuntu+debian+USB转串口+kermit

    当前的环境是:在Win7笔记本主机上安装VirtualBox+Ubuntu12_04,串口使用USB转串口 如果使用的虚拟机是VirtualBox: 如果使用的虚拟机是Vmware: 执行这步后,主机 ...

  10. DBA_SEGMENTS - 查看数据库对象所分配的物理存储空间

    SELECT SEGMENT_NAME,SEGMENT_TYPE,<code>TABLESPACE_NAME</code>,EXTENTS,BLOCKS,BYTES/1024/ ...