更新ViewModel属性时如何避免递归?

时间:2009-06-18 14:05:41

标签: c# wpf mvvm data-binding

在我的视图中,我有一个滑块和一个组合框

当我更改滑块时,我希望更改组合框。

当我更改组合框时,我希望滑块更改。

我可以尝试更新其中一个,但如果我尝试更新两者,我会收到 StackOverflow错误,因为一个属性会在无限循环中不断更新另一个属性。

我尝试过一个重新计算(),其中更新在一个地方完成,但仍会遇到递归问题。

如何在不进行递归的情况下让每个控件更新另一个?

在视图中

<ComboBox 
    ItemsSource="{Binding Customers}"
    ItemTemplate="{StaticResource CustomerComboBoxTemplate}"
    Margin="20"
    HorizontalAlignment="Left"
    SelectedItem="{Binding SelectedCustomer, Mode=TwoWay}"/>


<Slider Minimum="0" 
        Maximum="{Binding HighestCustomerIndex, Mode=TwoWay}" 
        Value="{Binding SelectedCustomerIndex, Mode=TwoWay}"/>
ViewModel中的

#region ViewModelProperty: SelectedCustomer
private Customer _selectedCustomer;
public Customer SelectedCustomer
{
    get
    {
        return _selectedCustomer;
    }

    set
    {
        _selectedCustomer = value;
        OnPropertyChanged("SelectedCustomer");
        SelectedCustomerIndex = _customers.IndexOf(_selectedCustomer);
    }
}
#endregion

#region ViewModelProperty: SelectedCustomerIndex
private int _selectedCustomerIndex;
public int SelectedCustomerIndex
{
    get
    {
        return _selectedCustomerIndex;
    }

    set
    {
        _selectedCustomerIndex = value;
        OnPropertyChanged("SelectedCustomerIndex");
        SelectedCustomer = _customers[_selectedCustomerIndex];
    }
}
#endregion

2 个答案:

答案 0 :(得分:7)

尝试设置函数:

public int SelectedCustomerIndex
{
    get
    {
        return _selectedCustomerIndex;
    }

    set
    {
        if (value != _selectedCustomerIndex)
        {
         _selectedCustomerIndex = value;
         OnPropertyChanged("SelectedCustomerIndex");
         SelectedCustomer = _customers[_selectedCustomerIndex];
        }
    }
}

仅在值发生实际变化时触发事件。这样,第二次调用具有相同值的set属性不会导致另一个更改事件。

当然,你必须为其他财产做这件事。

答案 1 :(得分:0)

两个属性都是相互调用的,因此是递归。与绑定无关。正确的方法是在属性更改时相互更改并触发两个属性的更改通知:

    public Customer SelectedCustomer
    {
        get
        {
            return _selectedCustomerIndex;
        }

        set
        {
            _selectedCustomer = value;
            _selectedCustomerIndex = _customers.IndexOf(value);

            OnPropertyChanged("SelectedCustomer");
            OnPropertyChanged("SelectedCustomerIndex");
        }
    }

    public int SelectedCustomerIndex
    {
        get
        {
            return _selectedCustomerIndex;
        }

        set
        {
            _selectedCustomerIndex = value;
            _selectedCustomer = _customers[_selectedCustomerIndex];

            OnPropertyChanged("SelectedCustomer");
            OnPropertyChanged("SelectedCustomerIndex");
        }
    }