Winform数据绑定

news2024/12/29 14:50:02

简介#

在C#中提起控件绑定数据,大部分人首先想到的是WPF,其实Winform也支持控件和数据的绑定。

Winform中的数据绑定按控件类型可以分为以下几种:

  • 简单控件绑定
  • 列表控件绑定
  • 表格控件绑定

绑定基类#

绑定数据类必须实现INotifyPropertyChanged接口,否则数据类属性的变更无法实时刷新到界面,但可以从界面刷新到类。
为了方便,我们设计一个绑定基类:


    
    
  1. / / / <summary >
  2. / / / 数据绑定基类
  3. / / / < /summary >
  4. public abstract class BindableBase : INotifyPropertyChanged
  5. {
  6. public event PropertyChangedEventHandler PropertyChanged;
  7. protected bool SetProperty <T >( ref T field, T newValue, [CallerMemberName] string propertyName = null)
  8. {
  9. if (!EqualityComparer <T >. Default.Equals(field, newValue))
  10. {
  11. field = newValue;
  12. PropertyChanged?. Invoke(this, new PropertyChangedEventArgs(propertyName));
  13. return true;
  14. }
  15. return false;
  16. }
  17. }

需要绑定的数据类继承绑定基类即可:


    
    
  1. / / / <summary >
  2. / / / 数据类
  3. / / / < /summary >
  4. public class Data : BindableBase
  5. {
  6. private int id = 0;
  7. private string name = string.Empty;
  8. public int ID { get = > id; set = > SetProperty( ref id, value); }
  9. public string Name { get = > name; set = > SetProperty( ref name, value); }
  10. }

功能扩展#

主要为绑定基类扩展了以下两个功能:

  • 获取属性的Description特性内容
  • 从指定类加载属性值,对象直接赋值是赋值的引用,控件绑定的数据源还是之前的对象

这两个功能不属于绑定基类的必要功能,但可以为绑定提供方便,所以单独放在扩展方法类里面。
代码如下:


    
    
  1. / / / <summary >
  2. / / / 数据绑定基类的扩展方法
  3. / / / < /summary >
  4. public static class BindableBaseExtension
  5. {
  6. / / / <summary >
  7. / / / 获取属性的描述,返回元组格式为 Item 1:描述信息 Item 2:属性名称
  8. / / / < /summary >
  9. / / / <param name = "type" > < /param >
  10. / / / <returns > < /returns >
  11. public static Tuple < string, string >[] GetDescription(this BindableBase bindData)
  12. {
  13. var proAry = bindData.GetType().GetProperties();
  14. var desAry = new Tuple < string, string >[proAry. Length];
  15. string desStr;
  16. for (int i = 0; i < proAry. Length; i + +)
  17. {
  18. var attrs = (DescriptionAttribute[])proAry[i].GetCustomAttributes(typeof(DescriptionAttribute), false);
  19. desStr = proAry[i].Name;
  20. foreach (DescriptionAttribute attr in attrs)
  21. {
  22. desStr = attr.Description;
  23. }
  24. desAry[i] = Tuple.Create(desStr, proAry[i].Name);
  25. }
  26. return desAry;
  27. }
  28. / / / <summary >
  29. / / / 加载同类型指定对象的属性值,如果当前属性值或目标属性值为 null则不执行赋值操作
  30. / / / < /summary >
  31. / / / <param name = "data" > < /param >
  32. public static void Load(this BindableBase source, BindableBase dest)
  33. {
  34. if ( source = = null || dest = = null)
  35. {
  36. / /不执行操作
  37. return;
  38. }
  39. Type type = source.GetType();
  40. if ( type ! = dest.GetType())
  41. {
  42. throw new ArgumentNullException( "参数类型不一致");
  43. }
  44. var proAry = type.GetProperties();
  45. for (int i = 0; i < proAry. Length; i + +)
  46. {
  47. var proType = proAry[i].PropertyType;
  48. if (proType.IsSubclassOf(typeof(BindableBase)))
  49. {
  50. / /检测到内部嵌套的绑定基类,建议不处理直接跳过,这种情况应该单独处理内嵌对象的数据加载
  51. / /var childData = (BindableBase)(proAry[i].GetValue( source));
  52. / /childData.Load((BindableBase)(proAry[i].GetValue(dest)));
  53. }
  54. else
  55. {
  56. proAry[i].SetValue( source, proAry[i].GetValue(dest));
  57. }
  58. }
  59. }
  60. }

简单控件绑定#

简单属性绑定是指某对象属性值和某控件属性值之间的简单绑定,需要了解以下内容:

  • Control.DataBindings 属性:代表控件的数据绑定的集合。
  • Binding 类:代表某对象属性值和某控件属性值之间的简单绑定。

使用方法如下:


    
    
  1. Data data = new Dat a() { ID = 1,Name = "test"};
  2. / /常规绑定方法
  3. textBox 1.DataBindings. Add( "Text", data, "ID");
  4. / /使用这种方式避免硬编码
  5. textBox 2.DataBindings. Add( "Text", data, nameof( data.Name));

注:这种绑定会自动处理字符串到数据的类型转换,转换失败会自动恢复原值。

列表控件绑定#

列表控件绑定主要用于 ListBox 与 ComboBox 控件,它们都属于 ListControl 类的派生类。ListControl 类为 ListBox 类和 ComboBox 类提供一个共同的成员实现方法。

注:CheckedListBox 类派生于 ListBox 类,不再单独说明。

使用列表控件绑定前,需要了解以下内容:

  • ListControl.DataSource 属性:获取或设置此 ListControl 的数据源,值为实现 IList 或 IListSource 接口的对象,如 DataSet 或 Array。

  • ListControl.DisplayMember 属性:获取或设置要为此 ListControl 显示的属性,指定 DataSource 属性指定的集合中包含的对象属性的名称,默认值为空字符串("")。

  • ListControl.ValueMember 属性:获取或设置属性的路径,它将用作 ListControl 中的项的实际值,表示 DataSource 属性值的单个属性名称,或解析为最终数据绑定对象的属性名、单个属性名或句点分隔的属性名层次结构, 默认值为空字符串("")。

注:最终的选中值只能通过ListControl.SelectedValue 属性获取,目前还没找到可以绑定到数据的方法。

绑定BindingList集合#

BindingList是一个可用来创建双向数据绑定机制的泛型集合,使用方法如下:


    
    
  1. BindingList < Data > list = new BindingList < Data >();
  2. list. Add(new Dat a() { ID = 1, Name = "name1" });
  3. list. Add(new Dat a() { ID = 2, Name = "name2" });
  4. comboBox 1.DataSource = list;
  5. comboBox 1.ValueMember = "ID";
  6. comboBox 1.DisplayMember = "Name";

注:如果使用List泛型集合则不支持双向绑定。同理,如果Data没有继承绑定基类,则属性值的变更也不会实时更新到界面。

绑定DataTable表格#

DataTable支持双向绑定,使用方法如下:


    
    
  1. DataTable dt = new DataTable();
  2. DataColumn[] dcAry = new DataColumn[]
  3. {
  4. new DataColumn( "ID"),
  5. new DataColumn( "Name")
  6. };
  7. dt. Columns.AddRange(dcAry);
  8. dt.Rows. Add( 1, "name1Dt");
  9. dt.Rows. Add( 2, "name2Dt");
  10. comboBox 1.DataSource = dt;
  11. comboBox 1.ValueMember = "ID";
  12. comboBox 1.DisplayMember = "Name";

绑定BindingSource源#

BindingSource 类封装窗体的数据源,旨在简化将控件绑定到基础数据源的过程,详细内容可查看 BindingSource 组件概述。

有时候数据类型可能没有实现INotifyPropertyChanged接口,并且这个数据类型我们还修改不了,这种情况就只能使用BindingSource来将控件绑定到数据了。

假设Data类没有继承BindableBase,绑定方法如下:


    
    
  1. List < Data > list = new List < Data >();
  2. list. Add(new Dat a() { ID = 1, Name = "name1" });
  3. list. Add(new Dat a() { ID = 2, Name = "name2" });
  4. BindingSource bs = new BindingSource();
  5. bs.DataSource = list;
  6. comboBox 1.DataSource = bs;
  7. comboBox 1.ValueMember = "ID";
  8. comboBox 1.DisplayMember = "Name";

关键是下面的步骤,改变集合内容时手动触发变更:


    
    
  1. / /单项数据变更
  2. list[ 0].Name = "test";
  3. bs.ResetItem( 0);
  4. / /添加数据项
  5. list. Add(new Dat a() { ID = 3, Name = "name3" });
  6. bs.ResetBindings( false);
  7. / /在BindingSource上添加或使用BindingList列表,则可以不用手动触发变更通知
  8. bs. Add(new Dat a() { ID = 4, Name = "name4" });

表格控件绑定#

绑定DataTable#

方法如下:


    
    
  1. DataColumn c 1 = new DataColumn( "ID", typeof( string));
  2. DataColumn c 2 = new DataColumn( "名称", typeof( string));
  3. dt. Columns. Add(c 1);
  4. dt. Columns. Add(c 2);
  5. dt.Rows. Add( 11, 22);
  6. / /禁止添加行,防止显示空白行
  7. dataGridView 1.AllowUserToAddRows = false;
  8. / /选择是否自动创建列
  9. dataGridView 1.AutoGenerateColumns = true;
  10. dataGridView 1.DataSource = dt.DefaultView;

绑定BindingList#

方法如下:


    
    
  1. / /填充数据
  2. BindingList < Data > dataList = new BindingList < Data >();
  3. for (int i = 0; i < 5; i + +)
  4. {
  5. dataList. Add(new Dat a() { ID = i, Name = "Name" + i.ToString() });
  6. }
  7. / /禁止添加行,防止显示空白行
  8. dataGridView 1.AllowUserToAddRows = false;
  9. / /选择是否自动创建列
  10. dataGridView 1.AutoGenerateColumns = false;
  11. / /手动创建列
  12. var desAry = dataList[ 0].GetDescription();
  13. int idx = 0;
  14. foreach (var des in desAry)
  15. {
  16. idx = dataGridView 1. Columns. Add($ "column{idx}", des.Item 1); / / 手动添加某列
  17. dataGridView 1. Columns[idx].DataPropertyName = des.Item 2; / / 设置为某列的字段
  18. }
  19. / /绑定集合
  20. dataGridView 1.DataSource = dataList;
  21. / /集合变更事件
  22. dataList.ListChanged + = DataList_ListChanged;

注:上面的GetDescription()是绑定基类的扩展方法。

BindingList提供集合的变更通知,Data通过继承绑定基类提供属性值的变更通知。

UI线程全局类#

上面所有绑定的数据源都不支持非UI线程的写入,会引起不可预知的问题,运气好的话也不会报异常出来。
为了方便多线程情况下更新数据源,设计一个UIThread类封装UI线程SynchronizationContext的PostSend的操作,用来处理所有的UI更新操作,关于SynchronizationContext可以参考SynchronizationContext 综述。

代码如下:


    
    
  1. / / / <summary >
  2. / / / UI线程全局类
  3. / / / < /summary >
  4. public static class UIThread
  5. {
  6. private static SynchronizationContext context;
  7. / / / <summary >
  8. / / / 同步更新UI控件的属性及绑定数据源
  9. / / / < /summary >
  10. / / / <param name = "act" > < /param >
  11. / / / <param name = "state" > < /param >
  12. public static void Send(Action < object > act, object state)
  13. {
  14. context. Send(obj = > { act(obj); }, state);
  15. }
  16. / / / <summary >
  17. / / / 同步更新UI控件的属性及绑定数据源
  18. / / / < /summary >
  19. / / / <param name = "act" > < /param >
  20. public static void Send(Action act)
  21. {
  22. context. Send(obj = > { act(); }, null);
  23. }
  24. / / / <summary >
  25. / / / 异步更新UI控件的属性及绑定数据源
  26. / / / < /summary >
  27. / / / <param name = "act" > < /param >
  28. / / / <param name = "state" > < /param >
  29. public static void Post(Action < object > act, object state)
  30. {
  31. context.Post(obj = > { act(obj); }, state);
  32. }
  33. / / / <summary >
  34. / / / 异步更新UI控件的属性及绑定数据源
  35. / / / < /summary >
  36. / / / <param name = "act" > < /param >
  37. public static void Post(Action act)
  38. {
  39. context.Post(obj = > { act(); }, null);
  40. }
  41. / / / <summary >
  42. / / / 在UI线程中初始化,只取第一次初始化时的同步上下文
  43. / / / < /summary >
  44. public static void Init()
  45. {
  46. if (context = = null)
  47. {
  48. context = SynchronizationContext.Current;
  49. }
  50. }
  51. }

直接在主界面的构造函数里面初始化即可:

UIThread.Init();

    
    

使用方法如下:


    
    
  1. Task. Run(() = >
  2. {
  3. / /同步更新UI
  4. UIThread. Send(() = > { dataList.RemoveAt( 0); });
  5. });
</article>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1541037.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Docker 安装 Nginx 容器,反向代理

Docker官方镜像https://hub.docker.com/ 寻找Nginx镜像 下载Nginx镜像 docker pull nginx #下载最新版Nginx镜像 (其实此命令就等同于 : docker pull nginx:latest ) docker pull nginx:xxx #下载指定版本的Nginx镜像 (xxx指具体版本号)检查当前所有Docker下载的镜像 docker…

Linux 创建交换空间

&#x1f680; 作者主页&#xff1a; 有来技术 &#x1f525; 开源项目&#xff1a; youlai-mall &#x1f343; vue3-element-admin &#x1f343; youlai-boot &#x1f33a; 仓库主页&#xff1a; Gitee &#x1f4ab; Github &#x1f4ab; GitCode &#x1f496; 欢迎点赞…

【C++】使用cppcheck检查C++代码

Cppcheck 是 C/C 代码的静态分析工具。它提供独特的代码分析来检测错误&#xff0c;并专注于检测未定义的行为和 危险的编码结构&#xff0c;即使它具有非标准语法&#xff08;在嵌入式项目中很常见&#xff09;。 关于静态分析 通过静态分析可以发现的错误类型包括&#xff…

Zabbix使用TimescaleDB数据库

一、前言 Zabbix 6.0 已发布很久&#xff0c;下个季度7.0应该会正式发布&#xff0c;但6.0也有许多新功能和新特性&#xff0c;这里介绍 6.0 配置 TimescaleDB&#xff0c;此安装配置方法可基本通用与其他版本。 二、TimescaleDB TimescaleDB 基于 PostgreSQL 数据库打造的一…

【Leetcode】2549. 统计桌面上的不同数字

文章目录 题目思路代码复杂度分析时间复杂度空间复杂度 结果总结 题目 题目链接&#x1f517; 给你一个正整数 n n n &#xff0c;开始时&#xff0c;它放在桌面上。在 1 0 9 10^9 109 天内&#xff0c;每天都要执行下述步骤&#xff1a; 对于出现在桌面上的每个数字 x &am…

Programming Abstractions in C阅读笔记:p331-p337

《Programming Abstractions in C》学习第79天&#xff0c;p331-p337&#xff0c;总计7页。 一、技术总结 /** File: stack.h* -------------* This interface defines an abstraction for stacks. In any* single application that uses this interface, the values in* the…

2024/3/24 LED点阵屏

显示原理&#xff1a; 类似矩阵键盘&#xff0c;逐行or逐列扫描 74HC595是串行 寄存器 感觉就是三转八寄存器 并行&#xff1a;同时输出&#xff1b;串行&#xff1a;一位一位输出 先配置74HC595&#xff0c;重新进行位声明 sbit RCKP3^5; //RCLK sbit SCKP3^6; …

ai问答机器人是什么?介绍这几款实用ai问答机器人

ai问答机器人是什么&#xff1f;随着人工智能技术的飞速发展&#xff0c;AI问答机器人已成为我们生活中不可或缺的一部分。它们能够智能地解答各种问题&#xff0c;提供便捷的服务&#xff0c;极大地提升了用户体验。本文将带你了解AI问答机器人的基本概念&#xff0c;并介绍几…

30-函数(上)

30-1 函数是什么 在计算机科学中&#xff0c;子程序是一个大型程序中的某部分代码&#xff0c;由一个或多个语句块组成。它负责完成某项特定任务&#xff0c;而且相较于其他代码&#xff0c;具备相对的独立性。 一般会有输入参数并有返回值&#xff0c;提供对过程的封装和细节…

jenkins配置源码管理的git地址时,怎么使用不了 credential凭证信息

前提 Jenkins使用docker部署 问题 &#xff08;在jenlins中设置凭证的方式&#xff09;在Jenkins的任务重配置Git地址&#xff0c;并且设置了git凭证,但是验证不通过&#xff0c;报错; 无法连接仓库&#xff1a;Command "git ls-remote -h -- http://192.1XX.0.98:X02/…

梅林生态第一个Defi借贷协议零撸教程

简介&#xff1a;Avalon Finance是第一个基于Merlin Chain的去中心化借贷协议。它包括三个关键组成部分&#xff1a;超额抵押借贷、与借贷相关的衍生品交易和基于借贷的算法稳定币。 相关概念&#xff1a;梅林生态、Defi 融资信息&#xff1a;项目于3月15日完成种子轮融资&am…

IntelliJ IDEA集成git配置账号密码

1 背景说明 刚使用IDEA,本地也安装Git,在提交和拉取代码的时候,总提示登录框,而且登录框还不能输入账号密码,只能输入登录Token。如下: 从而无法正常使用IDEA的Git功能,很苦恼。 2 解决方法 2.1 安装Git 进入官网地址 https://git-scm.com/,点击下载: 浏览器直接…

数据结构——栈和队列的表示与实现详解

目录 1.栈的定义与特点 2.队列的定义与特点 3.案例引入 4.栈的表示和操作的实现 1.顺序栈的表示 代码示例&#xff1a; 2.顺序栈的初始化 代码示例&#xff1a; 3.判断栈是否为空 代码示例&#xff1a; 4.求顺序栈长度 代码示例&#xff1a; 5.清空顺序栈 …

官宣|阿里巴巴捐赠的 Flink CDC 项目正式加入 Apache 基金会

摘要&#xff1a;本文整理自阿里云开源大数据平台徐榜江 (雪尽)&#xff0c;关于阿里巴巴捐赠的 Flink CDC 项目正式加入 Apache 基金会&#xff0c;内容主要分为以下四部分&#xff1a; 1、Flink CDC 新仓库&#xff0c;新流程 2、Flink CDC 新定位&#xff0c;新玩法 3、Flin…

taro框架之taro-ui中AtSwipeAction的使用

题记&#xff1a;所需效果&#xff1a;滑动删除 工作进程 官网文档代码 <AtSwipeAction options{[{text: 取消,style: {backgroundColor: #6190E8}},{text: 确认,style: {backgroundColor: #FF4949}} ]}><View classNamenormal>AtSwipeAction 一般使用场景</…

Vue 3 里的 onMounted 怎么用?

疑问 最近&#xff0c;一直在学习 Vue 3&#xff0c;此前我不懂前端&#xff0c;也没写过 Vue 2&#xff0c;所以是从 0 开始学习 Vue 3 的。很多对普通人不是疑问的&#xff0c;在我这里也会不太清楚。 我在写项目的时候&#xff0c;常见的一种场景是这样的&#xff1a;页面…

Excel通过下拉菜单,显示不同图片

背景&#xff1a;有的时候需要通过更改下拉菜单来改变对应的 值/ 图片。 如果是数值的话就是我们常常用的Vlookup&#xff0c;这个可以很简单的实现这个功能。&#xff08;这个如果不知道请自行百度&#xff09; 但是如果是图片呢&#xff1f;这个就不常见了&#xff0c;那么…

41 物体检测和目标检测数据集【李沐动手学深度学习v2课程笔记】

目录 1. 物体检测 2. 边缘框实现 3.数据集 4. 小结 1. 物体检测 2. 边缘框实现 %matplotlib inline import torch from d2l import torch as d2ld2l.set_figsize() img d2l.plt.imread(../img/catdog.jpg) d2l.plt.imshow(img);#save def box_corner_to_center(boxes):&q…

企业计算机服务器中了mkp勒索病毒怎么办,mkp勒索病毒解密流程步骤

在网络技术飞速发展的今天&#xff0c;越来越多的企业走向了数字化办公模式&#xff0c;网络为企业的生产运营提高了效率&#xff0c;为企业带来了极大便利&#xff0c;但网络是一把双刃剑&#xff0c;在为人们提供便利的同时也会带来数据安全问题&#xff0c;网络数据安全一直…

NFS网络文件系统

目录 简介 案例 配置autofs自动挂载 客户端配置autofs 简介 NFS&#xff08;Network File System&#xff0c;网络文件系统&#xff09;是FreeBSD支持的文件系统中的一种&#xff0c;它允许网络中的计算机&#xff08;不同的计算机、不同的操作系统&#xff09;之间通过TCP/I…