Combobox出现System.Data.DataRowView的原因,以及指定ValueMember的时机问题

当使用Combobox控件时,出现SelectedValue的值为“System.Data.DataRowView”的原因有两个:在Combobox的DataSource不为空的情况下,要么是没有为ValueMember赋值,要么是赋值赋错了,这两种情况下系统就会把SelectedValue的默认值输出来(注意红色部分,如果DataSource为空,那么SelectedValue的值为null)。但有时即使你对ValueMember赋了正确的值也会出现这个问题,这里有一个赋值时机选择的问题,请看下面的示例:

首先,构造如下所示的一个窗体:

窗体有一个下拉框,名为Combobox1,还有一个文本框,名为TextBox1,功能很简单,当选择下拉框的某一项的时候,把该选择项对应的ValueMember的值显示到文本框中。下面是实现该功能的核心代码:

Code-1:

[c-sharp] view plaincopy

  1. namespace FrmForTest
  2. {
  3. public partial class Form1 : Form
  4. {
  5. public Form1()
  6. {
  7. InitializeComponent();
  8. this.InitialCombobox();
  9. }
  10. private void InitialCombobox()
  11. {
  12. DataTable table = new DataTable();
  13. DataColumn column;
  14. DataRow row;
  15. column = new DataColumn("Name");
  16. table.Columns.Add(column);
  17. column = new DataColumn("Value");
  18. table.Columns.Add(column);
  19. for (int i = 0; i < 5; i++)
  20. {
  21. row = table.NewRow();
  22. row["Name"] = "Test" + i;
  23. row["Value"] = i.ToString();
  24. table.Rows.Add(row);
  25. }
  26. this.comboBox1.DataSource = table;
  27. this.comboBox1.DisplayMember = "Name";
  28. this.comboBox1.ValueMember = "Value";
  29. }
  30. private void comboBox1_SelectedIndexChanged(object sender, EventArgs e)
  31. {
  32. this.textBox1.Text = this.comboBox1.SelectedValue.ToString();
  33. }
  34. }
  35. }

运行上面的代码,窗体初始化完毕后输出的结果如下:

文本框显示的是“System.Data.DataRowView”,前面说过,只有在没有为Combobox指定ValueMember的值或指定错的情况下才会是该字符串,可程序已经正确指定了ValueMember为什么还是这个结果呢?我试着选择其他的下拉项,发现显示的结果变成了预想的值,原因何在?

仔细观察程序发现原因在于ValueMember赋值的时机,上面的代码当执行完DataSource赋值语句之后,触发了SelectedIndexChanged事件,而此时还未指定ValueMember,所以窗体初次初始化后显示的是“System.Data.DataRowView”,解决方法很简单:只需把DataSource赋值语句放到DisplayMember、ValueMember语句之后。

好,这个问题解决了,新的问题又来了,请看下面的描述:

突然想到Combobox还有一个事件是SelectedValueChanged,还是上面的程序Code-1,把SelectedIndexChanged替换为SelectedValueChanged,会是什么效果呢?试了一下,没有报错,并且窗体初始化完毕后显示的就是预期的值0。呵呵,又找到了一种解决方法。问题还没完,请接着往下看:

上面SelectedValueChanged的试验中,DataSource是在DisplayMember、ValueMember前面的,如果放到它们两个后面呢?继续尝试,这次报错了——“未将对象引用到对象的示例”,对程序进行跟踪,当执行完this.comboBox1.ValueMember = "Value"语句之后,程序转到了comboBox1_SelectedIndexChanged方法,在this.textBox1.Text = this.comboBox1.SelectedValue.ToString();语句处报错,SelectedValue的值为空,因为此时还没有给DataSource赋值,所以报这个错。

现在可以总结了,当使用SelectedIndexChanged时,ValueMember在DataSource前进行赋值,当使用SelectedValueChanged时,ValueMember在DataSource后进行赋值。

还有一个问题很有意思,如下所示:

this.comboBox1.DisplayMember = "Name";

this.comboBox1.ValueMember = "ValueError";

this.comboBox1.DataSource = table;

上面故意把ValueMember的值写错了,运行起来没有任何问题(是指没有报错,此时的SelectedValue值都为System.Data.DataRowView),然后我们对这三行代码的顺序做一下调整,如下:

this.comboBox1.DataSource = table;

this.comboBox1.DisplayMember = "Name";

this.comboBox1.ValueMember = "ValueError";

上面ValueMember的值依旧是错的,但是对其赋值操作放到了DataSource语句之后,运行程序,在ValueMember赋值语句处报错如下:“无法绑定到值成员”。

当指定了DataSource的值后再对ValueMember赋错,如果此时赋值赋错的话,运行时就会报错,如果是在DataSource语句之前对ValueMember赋值则不会报错,大家在应用的时候一定要注意。

时间: 2024-10-17 19:09:23

Combobox出现System.Data.DataRowView的原因,以及指定ValueMember的时机问题的相关文章

(转)Combobox出现System.Data.DataRowView的原因,以及指定ValueMember的时机问题

原文地址 http://blog.csdn.net/lubiaopan/article/details/5915774 当使用Combobox控件时,出现SelectedValue的值为“System.Data.DataRowView”的原因有两个:在Combobox的DataSource不为空的情况下,要么是没有为ValueMember赋值,要么是赋值赋错了,这两种情况下系统就会把SelectedValue的默认值输出来(注意红色部分,如果DataSource为空,那么SelectedValu

comboBox的id返回System.Data.DataRowView

关系到ComboBox的DataSource,DisplayMember和ValueMember属性的设置顺序的问题. ComboBox的DataSource属性为object类型,但是需要实现IListSource接口的类型. 当把DataTable赋值给该属性后,如果此时没有指定ComboBox的DisplayMember和ValueMember属性的话, 这时获得的ComboBox的SelectValue属性将是System.Data.DataRowView的一个实例. 据此推测DataT

报错:无法将类型&quot;System.Data.EntityState&quot;隐式转换为&quot;System.Data.Entity.EntityState&quot;

报错:无法将类型"System.Data.EntityState"隐式转换为"System.Data.Entity.EntityState" 原因一: using System.Collections.Generic;using System.Linq;using System.Text;using System.Threading.Tasks;using System.Data.Entity; 是否引用了System.Data.Entity   是则:db.Ent

System.Data.SqlClient.SqlError: 尚未备份数据库的日志尾部

SQL还原时出现下面的错误,System.Data.SqlClient.SqlError: 尚未备份数据库 "***" 的日志尾部.如果该日志包含您不希望丢失的工作,请使用 BACKUP LOG WITH NORECOVERY 备份该日志.请使用 RESTORE 语句的 WITH REPLACE 或 WITH STOPAT 子句来只覆盖该日志的内容. (Microsoft.SqlServer.Smo)原因分析:    这是因为在线还原的数据库在最后备份后又产生了新的日志,所以按照默认设

还原数据库备份文件时,关于“System.Data.SqlClient.SqlError:媒体集有2个媒体簇,但只提供了1个。必须提供所有成员”的处理方式

好久没写博客了,最近在做毕设的权限管理模块,今天在还原数据库文件时,遇到了“System.Data.SqlClient.SqlError:媒体集有2个媒体簇,但只提供了1个.必须提供所有成员”这个错误,百度了一下,找到了原因和解决方式. 原因分析:1.备份的时候,旧路径没有删除,我添加了一个新路径,就无意中设置成了两个备份路径,SQL就会根据两个备份路径把备份文件分成两个.2.还原的时候源数据必须把两个备份文件都加载上.3.还原的新路径要手工修改成实际SQL的data文件夹路径.4.如果有同名数

解决&quot;System.Data.OracleClient 需要 Oracle 客户端软件 8.1.7 或更高版本&quot;

在用VS2008+Oracle9做环境连接Oracle时候,在VS 开发服务器运行正常,但IIS服务器调试和部署会报错! IIS服务器报错:System.Data.OracleClient 需要 Oracle 客户端软件 8.1.7 或更高版本. 出错的原因: 1.虽然报的是需要安装客户端8.1.7及以上版本,实际是由于.NET账户没有访问Oracle\bin文件夹的权限 2.在 Windows Server 2003/2008 或Windows XP的 NTFS系统中提供了高级的访问安全性,F

C++: read SQL server data using System::Data::SqlClient

stdafx.h: // stdafx.h : include file for standard system include files, // or project specific include files that are used frequently, but // are changed infrequently #pragma once #using <mscorlib.dll> #using <System.dll> #using <System.Dat

EntityFramework 找不到方法:“Void System.Data.Entity.DbModelBuilder.RegisterEntityType

问题原因,EF当前版本没有该方法,将EF版本升级即可. 1.packages.config <package id="EntityFramework" version="6.1.3" targetFramework="net40" /> 2.app.config <configSections> <!-- For more information on Entity Framework configuration,

SharePoint &quot;System.Data.SqlClient.SqlException (0x80131904): Parameter &#39;@someColumn&#39; was supplied multiple times.“

最近在处理SharePoint Office365的相关开发的时候发现了这样一个奇怪的现象: 无法通过API更新Editor field,只要已更新就会throw Exception,由于是Office365的Exception,无法单单从Exception中分析出问题原因,而且奇怪的是新创建的List也存在同样的问题. 试了半天也没找到解决办法,单单看field的SchemaXml也没发现什么比较特殊的地方. 无奈之下save了一个list template,上传到本地的SharePoint