我有一个与ui线程一起运行的后台程序线程。现在在backgrounder_doWork方法中,我有一个数据表,我正在从数据库添加信息。我希望它绑定到ui线程中的datagridview以显示给用户。因此,当dataTable使用来自数据库的新信息进行更新时,datagridview应自动刷新以添加/减去从后台线程收到的任何新行信息。我怎样才能做到这一点?我试过这个:
private delegate void dGValueDelegate();
private void dGVValue()
{
BindingSource bSource = new BindingSource();
dtFailures.DataSource = bSource;
bSource.DataSource = dt;
}
其中dt是类级变量。在backgrounder_dowork方法内部,在开始时我调用dGVVAlue方法然后在后台信息被添加到后台线程中的数据表中。然而,datagridview不会显示......
答案 0 :(得分:9)
以下代码适合您:
private delegate void SetDGVValueDelegate(BindingList<Something> items);
void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
{
// Some call to your data access layer to get dt
SetDGVValue(DataTable dt)
}
private void SetDGVValue(DataTable dt)
{
if (dataGridView1.InvokeRequired)
{
dataGridView1.Invoke(new SetDGVValueDelegate(SetDGVValue), dt);
}
else
{
dataGridView1.DataSource = dt;
}
}
在你的问题中你提到使用BindingSource,它没有Invoke方法 - 如果你有一个绑定源,你可以改为在表单上调用:
// On the form
public void SetBindingSourceDataSource(object newDataSource)
{
if (InvokeRequired)
Invoke(new Action<object>(SetBindingSourceDataSource), newDataSource);
else
this.bindingSource.DataSource = newDataSource;
}
您也可以使用Lamda表达式在一行中执行此操作:
void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
{
// Some call to your data access layer to get dt
dataGridView1.Invoke((Action)(() => dataGridView1.DataSource = dt));
}
答案 1 :(得分:3)
此外,使用DataGridView.Invoke(Delegate)方法将允许您确保在其所属的线程中对datagridview进行更改。