在WPF中,构造函数中设置DataContext=this与绑定到{RelativeSource Self}之间的区别是什么?

5
下面的代码可以正常工作:
AskWindow.xaml:
<Window
    x:Class='AskWPF.AskWindow'
    xmlns:x='http://schemas.microsoft.com/winfx/2006/xaml'
    xmlns='http://schemas.microsoft.com/winfx/2006/xaml/presentation'
    >

<DataGrid ItemsSource="{Binding SimpleItems}" />

</Window>

AskWindow.xaml.cs:

namespace AskWPF {

public class SimpleRow {
    private string firstColumn;
    private string secondColumn;

    public SimpleRow(string first, string second) {
        firstColumn = first;
        secondColumn = second;
    }

    public string FirstColumn {
        get { return firstColumn; }
        set { firstColumn = value; }
    }

    public string SecondColumn {
        get { return secondColumn; }
        set { secondColumn = value; }
    }
}

public partial class AskWindow : Window {

    private ObservableCollection<SimpleRow> simpleItems;

    public AskWindow() {
        InitializeComponent();
        DataContext = this;

        simpleItems = new ObservableCollection<SimpleRow>();
        simpleItems.Add(new SimpleRow("row 0, column 0", "row 0, column 1"));
        simpleItems.Add(new SimpleRow("row 1, column 0", "row 1, column 1"));
    }

    public ObservableCollection<SimpleRow> SimpleItems {
        get { return simpleItems; }
    }
}

}

但是,如果在Window标签中设置DataContext ='{Binding RelativeSource={RelativeSource Self}}'并注释掉DataContext=this这一行,我们将得到一个空窗口。为什么?

AskWindow.xaml:

<Window .... DataContext='{Binding RelativeSource={RelativeSource Self}}'>

    <DataGrid ItemsSource="{Binding SimpleItems}" />

</Window>

AskWindow.xaml.cs:

...
public AskWindow() {
    InitializeComponent();
    // DataContext = this;

    simpleItems = new ObservableCollection<SimpleRow>();
    simpleItems.Add(new SimpleRow("row 0, column 0", "row 0, column 1"));
    simpleItems.Add(new SimpleRow("row 1, column 0", "row 1, column 1"));
}
...
3个回答

1
我怀疑这与绑定的评估方式和时间有关。在后一种情况下,我认为绑定可能会在集合属性的值仍为空的情况下检索该值,然后您更改该属性(通过设置字段)而不对受影响的属性触发任何更改通知。
建议将InitializeComponent调用移动到构造函数的末尾或至少先设置字段。
通常我使用只读字段并立即初始化它:
private readonly ObservableCollection<Data> collection =
    new ObservableCollection<Data>();
public ObservableCollection<Data> Collection { get { return collection ; } }

1
这是我的猜测。在两种情况下,你的 Collection 在某一时刻为空。确切地说,在 InitializeComponent 之后。此时,初始数据绑定获取到了数据,但没有数据上下文。现在通过设置 DataContext,你的属性被引发,与之相关的每个绑定也被失效并刷新。这是我猜测的部分,它能够工作的原因是 ItemsSource 的绑定被推迟了,因此在下一行中只需设置集合即可正常工作。

简而言之:设置 Datacontext 将重新触发绑定。但在相对源示例中,你的绑定从一开始就起作用了,但集合为空,你从未告诉 WPF 重新获取绑定。如果直接初始化集合,它应该可以正常工作。


我对这种行为感到惊讶。看起来如果在调用InitializeComponent()时DataContext为空,则绑定将被延迟;如果DataContext不为空,则会绑定到对象的具体实例。 - andrey_t

0

实际上,绑定是正确的,而且它也起作用。为了更新屏幕,绑定必须接收到某些更改的通知。绑定首先进行评估,然后监听通知。在您的第二个版本中,当运行InitializeComponent时,绑定首先进行评估,但此时没有任何值,因此您看不到任何内容。之后创建值,但由于未发送任何通知,绑定不会重新评估。

因此,一个解决方案是在InitializeComponent之前初始化集合。

...
private ObservableCollection<SimpleRow> simpleItems = new ObservableCollection<SimpleRow>();
...

另一个解决方案将是愚蠢的,过度通知绑定有关某些内容已更改。

只是一个注意事项,可能这是为了学习目的,因为UI不应与模型混淆。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接