枚举与Combox的SelectedIndex绑定出现怪异问题的解决方案

原创 2013年12月04日 09:59:29

在Combox中有一系列的选项,其选项的SelectedIndex值对应了相应的枚举值,而这些枚举值会影响着后续的工作模式,所以就想着将Combox的SelectedIndex和枚举值进行双向绑定,代码如下

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

//Add
using System.Linq.Expressions;

namespace ComboxBinding
{
    public partial class FormMain : Form
    {
        public FormMain()
        {
            InitializeComponent();
        }

        private BindingDataProperyty _bindingDataProperty = null;

        private void FormMain_Load(object sender, EventArgs e)
        {
            for (int index = 0; index < 10; index++)
            {
                comboBoxData.Items.Add(index.ToString());
            }
            
            _bindingDataProperty = new BindingDataProperyty();
            comboBoxData.DataBindings.Add("SelectedIndex",
                                        _bindingDataProperty,
                                        "Way_pro",
                                        false,
                                        DataSourceUpdateMode.OnPropertyChanged);
        }

        private void buttonShow_Click(object sender, EventArgs e)
        {
            MessageBox.Show(((WAYS)_bindingDataProperty.Way_pro).ToString());
        }
    }

    public enum WAYS
    {
        WAY0 = 0x00,
        WAY1 = 0x01,
        WAY2 = 0x02,
        WAY3 = 0x03,
        WAY4 = 0x04,
        WAY5 = 0x05,
        WAY6 = 0x06,
        WAY7 = 0x07,
        WAY8 = 0x08,
        WAY9 = 0x09
    }

    public class BindingDataProperyty : INotifyPropertyChanged
    { 
        #region Way Property
        private WAYS _way = 0;
        public WAYS Way_pro
        {
            get
            {
                return _way;
            }
            set
            {

                if (value == _way)
                {
                    return;
                }
                _way = value;
                NotifyPropertyChanged(() => Way_pro);
            }
        }
        #endregion

        #region Way NotifyPropertyChanged
        public void NotifyPropertyChanged<T>(Expression<Func<T>> property)
        {
            if (PropertyChanged == null)
            {
                return;
            }
            var memberExpression = property.Body as MemberExpression;
            if (memberExpression == null)
            {
                return;
            }

            PropertyChanged.Invoke(this, new PropertyChangedEventArgs(memberExpression.Member.Name));
        }

        public event PropertyChangedEventHandler PropertyChanged;
        #endregion
    }
}
代码正常运行,可是当我在Combox选择了项之后,要显示相应的工作方式时,怪异的事情发生了。请看下图1

图1


我明明在Combox上选择了4,结果一点Show之后,却弹出了WAY0,同时Combox却自动变成了0,这是怎么回事?这时不是应该弹出WAY4才对的吗?

网上搜了一通之后,没有看到合适的解决方案。思来想去,问题可能是出在枚举的绑定上,那就先别用枚举来定义Way_pro了,先用int来。修改的代码如下

    public class BindingDataProperyty : INotifyPropertyChanged
    {
        #region Way Property
        private int _way = 0;
        public int Way_pro
        {
            get
            {
                return _way;
            }
            set
            {

                if (value == _way)
                {
                    return;
                }
                _way = value;
                NotifyPropertyChanged(() => Way_pro);
            }
        }
        #endregion        

       ...
    }


代码中就是把Way_pro的类型由WAYS改成了int.下面是运行结果(图2)

图2


Combox选择了6,Show出来的是WAY6,结果达到了预期。

这么看来,还真的是枚举引起的。目的达到了,但是为什么会引起这个问题,还是没有明白,于是进行了进一步的测试。

将Way_pro的类型再改回成枚举WAYS,同时将初始值改成WAY9.修改的代码如下

    public class BindingDataProperyty : INotifyPropertyChanged
    {
        #region Way Property
        private WAYS _way = WAYS.WAY9;
        public WAYS Way_pro
        {
            get
            {
                return _way;
            }
            set
            {

                if (value == _way)
                {
                    return;
                }
                _way = value;
                NotifyPropertyChanged(() => Way_pro);
            }
        }
        #endregion        
         ...
    }

下面是运行结果(图3)

图3


可以看到Combox的值为9,正是我们所绑定的值WAY9对应的SelectedIndex。这么看来,从枚举到SelectedIndex的绑定是OK的。

再结合前面图1的结果,当我们选择了4之后,Show出来的结果却还是WAY0,同时Combox变成了0.这说明,双向绑定从SelectedIndex到枚举时出错了,导致我们在点击Show时,获取的是Way_pro的默认值(见图4),再次触发Way_pro,将Way_pro的默认值重新绑定到了Combox上,于是Combox又显示了默认值。

图4


那我们怎么证明从SelectedIndex到枚举的绑定出错了呢?我们知道,在双向绑定时,控件上的属性值发生改变时,会触发相应绑定属性的set方法,如SelectedIndex改变会触发Way_pro的set方法,如果不触发那就说明异常了(如类型转换失败引起的异常)。下面是两段代码debug的截图。

图5


图6


从图5中我们可以看到,Combox的默认是WAY9,对应的SelectedIndex是9,当我们在Combox选择了5之后,Way_pro的set方法并没有被触发。而从图6中可以看到,Combox选择了5之后,Way_pro的set方法触发了,并且值value=5也值进去了。这就说明图5的方式出现了异常,所以set方法没有触发,而这个异常是SelectedIndex隐式转换成WAYS时出错了。

如果只是进行单向绑定(也就是只改变Way_pro的值,Combox的SelectedIndex就会相应的改变),是可以直接将枚举与SelectedIndex绑定的。如图7

图7


所以综上,Combox的SelectedIndex与枚举可以直接单向绑定,但是不能直接双向绑定。如果要进行双向绑定,所绑定的属性必须定义成int类型,在取值时再强转成枚举类型。

补充:对于Combox的数据绑定需要使用SelectItem,同时绑定的源需要附一个初始值。如果绑定的是SelectValue或者SelectText,有可能会出问题,特别是直接将数据源赋值到Combox.DataSource中时,SelectValue会为空值。参考代码如下

 private void FormMain_Load(object sender, EventArgs e)
        {
            ...//使用前面的代码
	    _bindingDataProperty.Data_pro=1;//先赋一个初始值
            comboBoxData.DataBindings.Add("SelectedItem",
                                         _bindingDataProperty,
                                         "Data_pro",
                                         false,
                                         DataSourceUpdateMode.OnPropertyChanged);
	   
        }  

private void buttonShow_Click(object sender, EventArgs e)
        {           
            string info = string.Format("Data:{0}", _bindingDataProperty.Data_pro);
            MessageBox.Show(info);
        }

public class BindingDataProperyty : INotifyPropertyChanged
    {

     ...//使用前面的代码

        #region Data Property
        private string Data_ = "";
        public string Data_pro
        {
            get
            {
                return Data_;
            }
            set
            {

                if (value == Data_)
                {
                    return;
                }
                Data_ = value;
                NotifyPropertyChanged(() => Data_pro);
            }
        }
        #endregion

       ...//使用前面的代码
    }


版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

WPF的Combobox的ItemSource第二次绑定(更新)问题

http://s.yanghao.org/program/viewdetail.php?i=156654 WPF的Combobox的ItemSource第二次绑定(更新)问题 我在做w...

Combox默认项设置的问题及解决方案

在开发时,对于Combox控件,我们会对Items预置一些值,同时希望显示一个默认值。Combox中没有默认index这样的选项,于是会想着在程序加载时设置Combox的SelectedIndex为一...

ibatis分组查询结果的返回参数绑定问题解决方案

这次需求是从两个表里取出数据。最开始是在dao层中直接取出一张表中的所有记录,因为一张表对应一个实体类,这样取出所有记录,一些逻辑上的业务可以在程序中去处理,这样虽然导致代码量比较大,但是是比较容易理...

Dubbo服务主机IP没有绑定的坑(dubbo注册时出现主机上没有的IP的解决方案)

初次使用dubbo,在研发环境和测试环境测试没有问题,然后将服务上线,上线后,Dubbo服务端启动正常,客户端启动失败,并提示 Caused by: java.lang.IllegalStateEx...

silverlight combox 绑定 source 时, 处理空值问题。

参考 : http://www.codeproject.com/Articles/202523/Silverlight-Combobox    public class ComboBoxHel...

c#(winform)中自定义ListItem类方便ComboBox和ListBox添加项完全解决(DropDownList 和Combox 数据绑定的常用方法) .

刚开始用.net 的winform开发,发现好些控件都很难用,可能是不熟悉的原因吧,这不,一个给ComboBox添加项的问题就搞的我很头疼,我要同时给一个项添加名字和值,怎么都没法加,查了查资料,又自...
  • snihcel
  • snihcel
  • 2013年04月01日 17:00
  • 813

c#(winform)中自定义ListItem类方便ComboBox和ListBox添加项完全解决(DropDownList 和Combox 数据绑定的常用方法)

刚开始用.net 的winform开发,发现好些控件都很难用,可能是不熟悉的原因吧,这不,一个给ComboBox添加项的问题就搞的我很头疼,我要同时给一个项添加名字和值,怎么都没法加,查了查资料,又自...

DateTimePicker控件绑定可空DateTime值的怪异情况解决办法

场景: WinForms窗体上有一个DateTimePicker控件,ShowCheckBox设置为true,控件放在TabControl的非活动TabPage中。 这个DateTimePicke...
  • effun
  • effun
  • 2012年10月21日 16:13
  • 2809
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:枚举与Combox的SelectedIndex绑定出现怪异问题的解决方案
举报原因:
原因补充:

(最多只允许输入30个字)