为什么会覆盖.GetHashCode在WinForms中清除这些数据绑定值?

Rac*_*hel 14 c# mdi devexpress .net-3.5 winforms

我们遇到了一个奇怪的错误,我们在调试时遇到了问题.

我们有一个使用Microsoft CAB,DevExpress组件和.Net 3.5的MDI工作区.

如果用户在工作区中打开两个窗口,每个窗口包含UserControl绑定到两个单独的数据模型,然后最小化它们,则第一个最小化窗口是在第二个窗口最小化时清除它的绑定字段.

.Equals.GetHashCode数据模型的方法已被重写,使两者的数据模型被认为是相等的.如果我们改变它以使它们是唯一的,我们就不会得到这种行为.

这是一些显示问题的示例伪代码

var a = new MyWindow();
a.DataModel = new SomeClass(123);
a.ShowInMdiWorkspace();

var b = new MyWindow();
b.DataModel = new SomeClass(123);
b.ShowInMdiWorksace();

a.Minimize();

// If SomeClass.GetHashCode() is overwritten to consider two objects  
// as equal based on the value passed in, then the data bindings for A
// get cleared on this call. If SomeClass.GetHashCode is unique, then 
// this problem does not happen.
b.Minimize();
Run Code Online (Sandbox Code Playgroud)

这是第二个窗口最小化时的调用堆栈:

在此输入图像描述

EndEditSession()堆栈中的呼叫跟踪上面,它被调用EndEditSession第二个最小化的窗口,而由当时的堆栈跟踪不会越过[External Code]到的OnChange断点我已经设置,它发射的改变方法第一个窗口.

EndEditSession() 是我们已经实现的定制,看起来像这样

protected void EndEditSession()
{
    IBindingValue bv = null;

    if (_bindingValues == null)
        return;

    if (_data != null)
    {
        foreach (KeyValuePair<string, IBindingValue> kvp in _bindingValues)
        {
            bv = kvp.Value;
            if (bv.IsBindable)
                ((PropertyManager)bv.Component.BindingContext[_data]).EndCurrentEdit();
        }
    }

}
Run Code Online (Sandbox Code Playgroud)

_bindingValues当UserControl初始化其数据绑定时,将填充.关键字段是绑定控件的名称,值字段是存储控件本身,其名称,绑定值和默认值的自定义对象.bv.Component返回设置绑定的控件,在我的测试中是一个自定义的DevExpressLookupEdit

_data包含数据模型UserControl,我可以验证它是否设置为第二个窗口的实例.

我最初的想法是BindingContext共享,所以错误PropertyManager被归还,但我已经证实.BindingContext两个表格和控件是分开的.

是否有可能两个单独UserControl的数据绑定到数据模型的两个单独的实例副本会GetHashCode在重写方法时将其绑定混淆,以便两个对象被认为是相等的?

我对WinForms绑定系统的内部工作方式不太熟悉,或者对CAB的MDI工作空间的管理方式非常熟悉.

我的理论是,当第一个窗口最小化时,它将卸载控件以节省内存,然后当第二个窗口最小化管理绑定的内部哈希表时,错误地混淆并运行更新以从第一个最小化窗口获取数据(现在是空白的)并更新其数据源.这个理论有很多漏洞,但这是我唯一能想到的.

nem*_*Bu4 0

BindingContext对象不与任何其他对象共享其字段和属性,BindingContext因为它的字段和属性不是静态的。

但是,一个对象可以用于BindingContext多个控件。

在第一种情况下,如果多个控件具有相同的父控件并且没有自己的控件BindingContext,则BindingContext该控件的属性将返回Control.Parent(.Parent...).BindingContext对象。

在第二种情况下,可能是这样的:

var bindingContext = new BindingContext();
var a = new SomeControl();
var b = new SomeControl();
a.BindingContext = bindingContext;
b.BindingContext = bindingContext;
Run Code Online (Sandbox Code Playgroud)

第三种情况BindingContext可以用这样的方式覆盖。

我不知道你的情况发生了什么,所以我只能建议在初始化数据绑定之前执行类似的操作:

var a = new SomeControl();
var b = new SomeControl();
a.BindingContext = new BindingContext();
b.BindingContext = new BindingContext();
Run Code Online (Sandbox Code Playgroud)


如果这不能解决您的问题,那么您需要检查对象的填充_bindingValues。在填充该对象的过程中,可能会填充错误的值。

  • @Rachel 你能提供一个示例项目吗?似乎没有足够的信息来解决您的问题。我认为问题不在于“BindingContext”。问题可能出在您的“_bindingValues”对象中。 (2认同)