项目中经常用DataTable在内存中存储并操作数据,在进行报表开发的时候,报表的各种过滤功能用这个内存表可以大现身手,但最近在使用过程中却遇到一个奇怪的现象,现将该问题及处理方法记录一下。这是在做护士站的执行单的时候遇到过的,我使用DataTable的Select方法根据条件对内存表进行过滤,但是偶尔会出现非预期结果,如:
1 string strFilter = "处方号={0}"; 2 strFilter = string.Format(strFilter, 123456); 3 DataRow[] row = dsDrugList.Select(strFilter);
在dsDrugList中有处方号=123456的数据,但是最终的row没有得到处方号为123456,百思不得其解。
因此查阅MSDN:
获取 DataRow 对象的数组。
名称 |
说明 |
|
|
获取所有 DataRow 对象的数组。 |
|
|
按照主键顺序(如果没有主键,则按照添加顺序)获取与筛选条件相匹配的所有DataRow 对象的数组。 |
|
|
获取按照指定的排序顺序且与筛选条件相匹配的所有 DataRow 对象的数组。 |
|
|
获取与排序顺序中的筛选器以及指定的状态相匹配的所有 DataRow 对象的数组 |
重点看下一个参数的重载方法:Select(String):
按照主键顺序(如果没有主键,则按照添加顺序)获取与筛选条件相匹配的所有 DataRow 对象的数组。
微软自己的示例:
1 private void GetRowsByFilter() 2 { 3 DataTable table = DataSet1.Tables["Orders"]; 4 // Presuming the DataTable has a column named Date. 5 string expression; 6 expression = "Date > #1/1/00#"; 7 DataRow[] foundRows; 8 9 // Use the Select method to find all rows matching the filter. 10 foundRows = table.Select(expression); 11 12 // Print column 0 of each returned row. 13 for(int i = 0; i < foundRows.Length; i ++) 14 { 15 Console.WriteLine(foundRows[i][0]); 16 } 17 }
嗯,并没有什么特别之处,晕乎了半小时。。。
最终MSDN中该方法的备注说明引起了我的思考。
若要创建 filterExpression 参数,请使用与应用于 DataColumn 类的 Expression 用来创建筛选器的属性值相同的规则。
难道我传递进去的过滤表达式与DataTable中的“属性值规则”不一样?但会是什么不一样呢,因此询问大师兄,并将我的怀疑告之,大师兄看了后建议我将条件转换为字符串试试,试了下过不如其然。
由于我的列“处方号”在数据库中是一个number类型,因此我通过filterExpression传递进去的参数没有加入引号,修改为如下即可完全正确select出期望的数据了。
如下('{0}'占位符加单引号):
1 string strFilter = "处方号='{0}'"; 2 strFilter = string.Format(strFilter, 123456); 3 DataRow[] row = dsDrugList.Select(strFilter);