频繁绑定DataGridView的DataSource却不正常显示

在多线程程序中,频繁跨线程绑定DataGridView的DataSource导致显示异常,如程序卡死或数据显示不全。尝试禁用跨线程调用检查、使用委托等方式仍无法解决问题。最终发现,减少DataSource的绑定频率或改为手动绑定可以解决这个问题,手动绑定的代码示例如文中所述。

最近在写一个多线程程序,需要跨线程访问DataGridView,绑定其DataSource,而且由于线程几乎是每隔几秒都会重新绑定一次DataGridView的DataSource的,所以,遇到各种蛋疼的问题。

首先说一个最常见最容易想到的办法:

首先在主线程设置System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls = false;,随后去Designer.cs文件中将DataGridView的声明修改成public static 然后,在自定义的线程中直接使用datagridview1.DataSource =dt;然后结果是程序莫名的卡死。

上网一查,大家都说跨线程访问控件时这么禁用跨线程调用检查是不科学的,需要声明一个委托来访问,于是继续折腾写了如下代码

public delegate void SetDGVSource(DataTable dt);
        public static void SetDGVSourceFunction(DataTable dt)
        {

            if (dataGridView1.InvokeRequired)
            {
                SetDGVSource delegateSetSource = new SetDGVSource(SetDGVSourceFunction);
                dataGridView1.Invoke(delegateSetSource, new object[] { dt });
            }
            else
            {
                dataGridView1.DataSource = dt;
                dataGridView1.Columns[dataGridView1.Columns.Count - 1].Visible = false;//设置最后一列不可见

            }
        }

这样确实可以访问了,当时,发现当程序运行了分吧钟的样子,问题又莫名其妙的来了:莫名的崩溃。

后来继续查资料,在论坛看到有人说需要添加什么绑定,发现不靠谱(WinForm程序不需要,那是ASP.NET的),继续查阅,有人说需要将DataGridView的一个属性改一下,

dataGridView1.AutoGenerateColumns = true;

然后我就老老实实的添加了,然后出现的问题更加奇葩

屏幕上的datagridview依然是一片空白,貌似根本没有添加上,正当丧气之时,猛然间发现鼠标经过datagridview的空白区时,鼠标由指针变成了手型,这不是说明datagridview里面有东西么!!!!!果然,将鼠标在datagridview里面胡乱点击拖动,就看到datagridview的一行行就这么被我一拖,它就显示出来了,我拖动一行就显示一行。。。。

蛋疼无比啊!!!!

不过在进一步的Debug中,我发现,如果不是频繁的刷新datagridview的DataSource的话,目测他又是正常的。。。

后来继续慢慢摸索,发现,在不改变设置DataSource的绑定频率的情况下,如果改用手动绑定DataSource,他又神奇般的好了。。。。

代码如下:

datagridview1.Rows.Clear();  
foreach (DataRow dr in dt.Rows)  
{  
    datagridview1.Rows.Add(dr["姓名"], dr["年龄"]);  
} 

就这样居然就可以解决我的问题了!真是喜极而泣。。。

也许是datagridview里面的DataSource直接绑定有什么机制导致了它不能够频繁的更新吧。只能采用手动方式。当然了,类似于上面的方法,还可以使用下面的更加简洁的代码

datagridview1.DataSource =dt.Copy();

最终我的代码如下:

public static void SetDGVSourceFunction(DataTable dt)
        {

            if (dataGridView1.InvokeRequired)
            {
                SetDGVSource delegateSetSource = new SetDGVSource(SetDGVSourceFunction);
                dataGridView1.Invoke(delegateSetSource, new object[] { dt });
            }
            else
            {
                dataGridView1.DataSource = dt.Copy();
                //dataGridView1.Rows.Clear();
				//foreach(DataRow dr in dt.Rows)
				//	dataGridView.Rows.Add(dr["姓名"],dr["年龄"]);
                dataGridView1.AutoGenerateColumns = true;
                dataGridView1.Columns[dataGridView1.Columns.Count - 1].Visible = false;//设置最后一列不可见

            }
        }


然后再在其他文件的其他的线程中只需要调用这个函数即可,便可成功的频繁的绑定datagridview的DataSource!

评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值