BindingList<> ListChanged 事件

Posted

技术标签:

【中文标题】BindingList<> ListChanged 事件【英文标题】:BindingList<> ListChanged event 【发布时间】:2010-11-23 23:33:32 【问题描述】:

我有一个类的 BindingList 设置为 BindingSource 的 DataSource 属性,而后者又设置为 DataGridView 的 DataSource 属性。

1。 据我了解,对列表的任何添加都会触发 ListChanged 事件,该事件将通过 BindingSource 传播,然后传播到 DataGridView,DataGridView 将自行更新以显示更改。这将发生,因为事件已自动连接。 (是吗?)

当所有工作都在 UI 线程上完成时,这一切都很好,但是当从非 UI 线程创建和更改列表时,最终会在更新网格时发生跨线程异常。我可以理解为什么会发生这种情况,但不知道如何解决它......

2。 我很难理解的是,我应该在哪里最好地拦截 ListChanged 事件以尝试将事物编组到 UI 线程上?我猜我需要以某种方式引用 UI 线程来帮助做到这一点?

我已经阅读了很多关于此的帖子/文章,但我很苦恼,因为我不完全了解这里的工作机制。

一旦它们在列表中,我将永远不会更改任何项目,只会添加它们,并最初清除列表。

(我使用的是 .NET 2.0)

【问题讨论】:

【参考方案1】:

您可以扩展 BindingList 以使用 ISynchronizeInvoke(由 System.Windows.Forms.Control 实现)将事件调用编组到 UI 线程。

那么你需要做的就是使用新的列表类型并且所有的都被排序了。

public partial class Form1 : System.Windows.Forms.Form 

    SyncList<object> _List; 
    public Form1() 
        InitializeComponent();
        _List = new SyncList<object>(this);
    


public class SyncList<T> : System.ComponentModel.BindingList<T> 

    private System.ComponentModel.ISynchronizeInvoke _SyncObject;
    private System.Action<System.ComponentModel.ListChangedEventArgs> _FireEventAction;

    public SyncList() : this(null) 
    

    public SyncList(System.ComponentModel.ISynchronizeInvoke syncObject) 

        _SyncObject = syncObject;
        _FireEventAction = FireEvent;
    

    protected override void OnListChanged(System.ComponentModel.ListChangedEventArgs args) 
        if(_SyncObject == null) 
            FireEvent(args);
        
        else 
            _SyncObject.Invoke(_FireEventAction, new object[] args);
        
    

    private void FireEvent(System.ComponentModel.ListChangedEventArgs args) 
        base.OnListChanged(args);
    

【讨论】:

我理解它,但它仍然很神奇,并且可以做每个人都想做的事情。通过数据结构更新控件,无需担心调用。 @bulltorious 你可以提供赏金:)【参考方案2】:

    这种观点很公平。在幕后,CurrencyManager 和 Binding 等其他对象可确保在底层数据源更改时更新控件。

    将项目添加到数据绑定的 BindingList 会触发一系列事件,这些事件最终会尝试更新 DataGridView。由于 UI 只能从 UI 线程更新,您应该从 UI 线程通过 Control.Invoke 将项目添加到 BindingList。

我组装了一个快速示例,创建了一个带有 DataGridView、BindingSource 和 Button 的表单。

该按钮启动另一个线程,该线程模拟获取新项目以包含在 BindingList 中。

包含本身是通过 Control.Invoke 在 UI 线程中完成的。


    public partial class BindingListChangedForm : Form 
        BindingList<Person> people = new BindingList<Person>();
        Action<Person> personAdder;

        public BindingListChangedForm() 
            InitializeComponent();
            this.dataGridView1.AutoGenerateColumns = true;
            this.bindingSource1.DataSource = this.people;
            this.personAdder = this.PersonAdder;
        

        private void button1_Click(object sender, EventArgs e) 
            Thread t = new Thread(this.GotANewPersononBackgroundThread);
            t.Start();
        

        // runs on the background thread.
        private void GotANewPersononBackgroundThread() 
            Person person = new Person  Id = 1, Name = "Foo" ;

            //Invokes the delegate on the UI thread.
            this.Invoke(this.personAdder, person);
        

        //Called on the UI thread.
        void PersonAdder(Person person) 
            this.people.Add(person);
        
    

    public class Person 
        public int Id  get; set; 
        public string Name  get; set; 
    

【讨论】:

阿尔弗雷德,谢谢你的清晰例子。但是,假设我有一个在另一个线程上创建的类,它创建并添加到该线程的列表中。要将数据绑定到网格,我需要对 UI 线程的引用,是吗?我不知道在这里做什么是最好的。也许通过类构造函数传入对表单的引用,并以这种方式做某事? 1.是的。 2. 通过 Form 的 构造函数或其他成员将对象的引用传递给表单。 安迪,对象通常与线程无关,控件是一个例外。它是在线程上执行的代码。

以上是关于BindingList<> ListChanged 事件的主要内容,如果未能解决你的问题,请参考以下文章

BindingList<T>.Sort() 表现得像 List<T>.Sort()

是啥导致 BindingList<T> 中的 ListChangedType.ItemMoved ListChange 事件?

交换 BindingList<SomeClass> 元素需要很多时间,为啥会这样,我该怎么办?

从 BindingList 中删除重复项

List<T> vs BindingList<T> 优点/缺点

使用BindingList来实现DataGridview数据源为list时的动态增删改