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

时间:2013-09-10 08:45:53

标签: c# wpf xaml datacontext relativesource

下一代码按预期工作:

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 个答案:

答案 0 :(得分:1)

我怀疑它与如何以及何时评估某些类型的绑定有关。在后一种情况下,我认为绑定可以检索collection属性的值,同时它仍为null,然后您更改属性(通过设置字段)而不触发受影响属性的任何更改通知。

建议将InitializeComponent调用移到构造函数的末尾或至少预先设置字段。

通常我使用只读字段并立即初始化它:

private readonly ObservableCollection<Data> collection =
    new ObservableCollection<Data>();
public ObservableCollection<Data> Collection { get { return collection ; } }

答案 1 :(得分:1)

这是我的猜测。在这两种情况下,您的Collection都是空的。确切地说,在InitializeComponent之后。此时,初始数据绑定获取数据,但没有datacontext。现在,通过设置DataContext,您的属性将被引发,并且与之相关的每个绑定都会失效并刷新。这是我的猜测部分,它起作用的原因是延迟了对ItemsSource的绑定,因此它只是在下一行设置集合。

简而言之:设置Datacontext将重新绑定绑定。但是在您的RelativeSource示例中,您的绑定从一开始就起作用,但该集合为null,您从未告诉wpf重新绑定绑定。 如果您直接初始化您的收藏,它应该可以正常工作。

答案 2 :(得分:0)

实际上绑定是正确的,它也有效。为了更新屏幕,绑定必须接收更改某些内容的通知。绑定首先评估然后侦听通知。在第二个版本中,绑定首先计算InitializeComponent运行的时间,但此时没有值,因此您什么也看不到。之后会创建值,但绑定不会重新评估,因为没有发送通知。

所以是的,一个解决方案是在InitializeComponent之前初始化集合。

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

另一个解决方案是愚蠢和过度杀戮,通知绑定事情发生了变化。

请注意,这可能是出于学习目的,因为UI不应与模型混淆。