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; }
    }
}

}

但是如果DataContext='{Binding RelativeSource={RelativeSource Self}}'在 Window 标签和注释行中设置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"));
}
...
4

3 回答 3

1

我怀疑这与评估某些类型的绑定的方式和时间有关。在后一种情况下,我认为绑定可能会在它仍然为空时检索集合属性的值,然后您更改属性(通过设置字段)而不为受影响的属性触发任何更改通知。

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

通常我使用只读字段并立即对其进行初始化:

private readonly ObservableCollection<Data> collection =
    new ObservableCollection<Data>();
public ObservableCollection<Data> Collection { get { return collection ; } }
于 2013-09-10T08:52:43.130 回答
1

这是我的猜测。在这两种情况下,您的 Collection 都为空。准确地说是在 InitializeComponent 之后。此时初始数据绑定得到了数据,但没有数据上下文。现在通过设置 DataContext 你的属性被提升并且与之相关的每个绑定都被无效和刷新。这是我的猜测部分,它起作用的原因是与 ItemsSource 的绑定被延迟,因此它可以在下一行中设置集合。

简而言之:设置 Datacontext 将重新触发绑定。但是在您的 RelativeSource 示例中,您的绑定从一开始就有效,但集合为空,并且您从未告诉 wpf 重新获取绑定。如果您直接初始化您的集合,它应该可以正常工作。

于 2013-09-10T09:29:00.657 回答
0

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

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

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

另一种解决方案将是愚蠢的,并且过于矫枉过正地通知绑定某些内容已更改。

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

于 2013-09-10T09:26:55.520 回答