Winform数据绑定
简介#
在C#中提起控件绑定数据,大部分人首先想到的是WPF,其实Winform也支持控件和数据的绑定。
Winform中的数据绑定按控件类型可以分为以下几种:
- 简单控件绑定
- 列表控件绑定
- 表格控件绑定
绑定基类#
绑定数据类必须实现INotifyPropertyChanged接口,否则数据类属性的变更无法实时刷新到界面,但可以从界面刷新到类。
为了方便,我们设计一个绑定基类:
- /// <summary>
- /// 数据绑定基类
- /// </summary>
- public abstract class BindableBase : INotifyPropertyChanged
- {
- public event PropertyChangedEventHandler PropertyChanged;
- protected bool SetProperty<T>(ref T field, T newValue, [CallerMemberName] string propertyName = null)
- {
- if (!EqualityComparer<T>.Default.Equals(field, newValue))
- {
- field = newValue;
- PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
- return true;
- }
- return false;
- }
- }
需要绑定的数据类继承绑定基类即可:
- /// <summary>
- /// 数据类
- /// </summary>
- public class Data : BindableBase
- {
- private int id = 0;
- private string name = string.Empty;
-
- public int ID { get => id; set => SetProperty(ref id, value); }
- public string Name { get => name; set => SetProperty(ref name, value); }
- }
功能扩展#
主要为绑定基类扩展了以下两个功能:
- 获取属性的Description特性内容
- 从指定类加载属性值,对象直接赋值是赋值的引用,控件绑定的数据源还是之前的对象
这两个功能不属于绑定基类的必要功能,但可以为绑定提供方便,所以单独放在扩展方法类里面。
代码如下:
- /// <summary>
- /// 数据绑定基类的扩展方法
- /// </summary>
- public static class BindableBaseExtension
- {
-
- /// <summary>
- /// 获取属性的描述,返回元组格式为 Item1:描述信息 Item2:属性名称
- /// </summary>
- /// <param name="type"></param>
- /// <returns></returns>
- public static Tuple<string, string>[] GetDescription(this BindableBase bindData)
- {
- var proAry = bindData.GetType().GetProperties();
- var desAry = new Tuple<string, string>[proAry.Length];
- string desStr;
- for (int i = 0; i < proAry.Length; i++)
- {
- var attrs = (DescriptionAttribute[])proAry[i].GetCustomAttributes(typeof(DescriptionAttribute), false);
- desStr = proAry[i].Name;
- foreach (DescriptionAttribute attr in attrs)
- {
- desStr = attr.Description;
- }
- desAry[i] = Tuple.Create(desStr, proAry[i].Name);
- }
- return desAry;
- }
-
-
- /// <summary>
- /// 加载同类型指定对象的属性值,如果当前属性值或目标属性值为null则不执行赋值操作
- /// </summary>
- /// <param name="data"></param>
- public static void Load(this BindableBase source, BindableBase dest)
- {
-
- if (source == null || dest == null)
- {
- //不执行操作
- return;
- }
- Type type = source.GetType();
- if (type != dest.GetType())
- {
- throw new ArgumentNullException("参数类型不一致");
- }
- var proAry = type.GetProperties();
- for (int i = 0; i < proAry.Length; i++)
- {
- var proType = proAry[i].PropertyType;
-
- if (proType.IsSubclassOf(typeof(BindableBase)))
- {
- //检测到内部嵌套的绑定基类,建议不处理直接跳过,这种情况应该单独处理内嵌对象的数据加载
- //var childData = (BindableBase)(proAry[i].GetValue(source));
- //childData.Load((BindableBase)(proAry[i].GetValue(dest)));
- }
- else
- {
- proAry[i].SetValue(source, proAry[i].GetValue(dest));
- }
- }
- }
- }
简单控件绑定#
简单属性绑定是指某对象属性值和某控件属性值之间的简单绑定,需要了解以下内容:
- Control.DataBindings 属性:代表控件的数据绑定的集合。
- Binding 类:代表某对象属性值和某控件属性值之间的简单绑定。
使用方法如下:
- Data data = new Data() { ID=1,Name="test"};
- //常规绑定方法
- textBox1.DataBindings.Add("Text", data, "ID");
- //使用这种方式避免硬编码
- textBox2.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是一个可用来创建双向数据绑定机制的泛型集合,使用方法如下:
- BindingList<Data> list = new BindingList<Data>();
- list.Add(new Data() { ID = 1, Name = "name1" });
- list.Add(new Data() { ID = 2, Name = "name2" });
-
- comboBox1.DataSource = list;
- comboBox1.ValueMember = "ID";
- comboBox1.DisplayMember = "Name";
注:如果使用List泛型集合则不支持双向绑定。同理,如果Data没有继承绑定基类,则属性值的变更也不会实时更新到界面。
绑定DataTable表格#
DataTable支持双向绑定,使用方法如下:
- DataTable dt = new DataTable();
- DataColumn[] dcAry = new DataColumn[]
- {
- new DataColumn("ID"),
- new DataColumn("Name")
- };
- dt.Columns.AddRange(dcAry);
- dt.Rows.Add(1, "name1Dt");
- dt.Rows.Add(2, "name2Dt");
-
- comboBox1.DataSource = dt;
- comboBox1.ValueMember = "ID";
- comboBox1.DisplayMember = "Name";
绑定BindingSource源#
BindingSource 类封装窗体的数据源,旨在简化将控件绑定到基础数据源的过程,详细内容可查看 BindingSource 组件概述。
有时候数据类型可能没有实现INotifyPropertyChanged接口,并且这个数据类型我们还修改不了,这种情况就只能使用BindingSource来将控件绑定到数据了。
假设Data类没有继承BindableBase,绑定方法如下:
- List<Data> list = new List<Data>();
- list.Add(new Data() { ID = 1, Name = "name1" });
- list.Add(new Data() { ID = 2, Name = "name2" });
-
- BindingSource bs = new BindingSource();
- bs.DataSource = list;
- comboBox1.DataSource = bs;
- comboBox1.ValueMember = "ID";
- comboBox1.DisplayMember = "Name";
关键是下面的步骤,改变集合内容时手动触发变更:
- //单项数据变更
- list[0].Name = "test";
- bs.ResetItem(0);
-
- //添加数据项
- list.Add(new Data() { ID = 3, Name = "name3" });
- bs.ResetBindings(false);
- //在BindingSource上添加或使用BindingList列表,则可以不用手动触发变更通知
- bs.Add(new Data() { ID = 4, Name = "name4" });
表格控件绑定#
绑定DataTable#
方法如下:
- DataColumn c1 = new DataColumn("ID", typeof(string));
- DataColumn c2 = new DataColumn("名称", typeof(string));
- dt.Columns.Add(c1);
- dt.Columns.Add(c2);
- dt.Rows.Add(11, 22);
-
- //禁止添加行,防止显示空白行
- dataGridView1.AllowUserToAddRows = false;
- //选择是否自动创建列
- dataGridView1.AutoGenerateColumns = true;
-
- dataGridView1.DataSource = dt.DefaultView;
绑定BindingList#
方法如下:
- //填充数据
- BindingList<Data> dataList = new BindingList<Data>();
- for (int i = 0; i < 5; i++)
- {
- dataList.Add(new Data() { ID = i, Name = "Name" + i.ToString() });
- }
-
-
- //禁止添加行,防止显示空白行
- dataGridView1.AllowUserToAddRows = false;
- //选择是否自动创建列
- dataGridView1.AutoGenerateColumns = false;
- //手动创建列
- var desAry = dataList[0].GetDescription();
- int idx = 0;
- foreach (var des in desAry)
- {
- idx = dataGridView1.Columns.Add($"column{idx}", des.Item1); // 手动添加某列
- dataGridView1.Columns[idx].DataPropertyName = des.Item2; // 设置为某列的字段
- }
- //绑定集合
- dataGridView1.DataSource = dataList;
- //集合变更事件
- dataList.ListChanged += DataList_ListChanged;
注:上面的GetDescription()是绑定基类的扩展方法。
BindingList提供集合的变更通知,Data通过继承绑定基类提供属性值的变更通知。
UI线程全局类#
上面所有绑定的数据源都不支持非UI线程的写入,会引起不可预知的问题,运气好的话也不会报异常出来。
为了方便多线程情况下更新数据源,设计一个UIThread类封装UI线程SynchronizationContext的Post、Send的操作,用来处理所有的UI更新操作,关于SynchronizationContext可以参考SynchronizationContext 综述。
代码如下:
- /// <summary>
- /// UI线程全局类
- /// </summary>
- public static class UIThread
- {
- private static SynchronizationContext context;
-
-
- /// <summary>
- /// 同步更新UI控件的属性及绑定数据源
- /// </summary>
- /// <param name="act"></param>
- /// <param name="state"></param>
- public static void Send(Action<object> act, object state)
- {
- context.Send(obj=> { act(obj); }, state);
- }
-
- /// <summary>
- /// 同步更新UI控件的属性及绑定数据源
- /// </summary>
- /// <param name="act"></param>
- public static void Send(Action act)
- {
- context.Send(obj => { act(); }, null);
- }
-
- /// <summary>
- /// 异步更新UI控件的属性及绑定数据源
- /// </summary>
- /// <param name="act"></param>
- /// <param name="state"></param>
- public static void Post(Action<object> act, object state)
- {
- context.Post(obj => { act(obj); }, state);
- }
-
- /// <summary>
- /// 异步更新UI控件的属性及绑定数据源
- /// </summary>
- /// <param name="act"></param>
- public static void Post(Action act)
- {
- context.Post(obj => { act(); }, null);
- }
-
-
- /// <summary>
- /// 在UI线程中初始化,只取第一次初始化时的同步上下文
- /// </summary>
- public static void Init()
- {
- if (context == null)
- {
- context = SynchronizationContext.Current;
- }
- }
- }
直接在主界面的构造函数里面初始化即可:
UIThread.Init();
使用方法如下:
- Task.Run(() =>
- {
- //同步更新UI
- UIThread.Send(() => { dataList.RemoveAt(0); });
- });
</article>
相关文章:

Winform数据绑定
简介# 在C#中提起控件绑定数据,大部分人首先想到的是WPF,其实Winform也支持控件和数据的绑定。 Winform中的数据绑定按控件类型可以分为以下几种: 简单控件绑定列表控件绑定表格控件绑定 绑定基类# 绑定数据类必须实现INotifyPropertyChanged…...

DeprecationWarning: currentThread() is deprecated, use current_thread() instead
解决方案: # auto_commit not getattr(threading.currentThread(), testing, False) #阙辉注释 auto_commit not getattr(threading.current_thread(), testing, False) #阙辉新增...
2024届 C++ 刷题 笔试强训 Day 03
选择题 01 以下程序的输出结果是() #include <stdio.h> void main() {char a[10] {1, 2, 3, 4, 5, 6, 7, 8, 9, 0}, *p;int i;i 8;p a i;printf("%s\n", p - 3); }A 6 B 6789 C ‘6’ D 789 题目解析: 题目中定义了一个…...

linux用git拉取我云端以及git处理冲突
拉取后切换一个跟云端分支(dev)一样的 git branch --set-upstream-toorigin/dev dev 之后就同步了 A在dev分支写了iii,提交 B在dev分支写了hhh,提交,冲突 怎么修改,B把云端的拉下来,随便改改就行...

Learn OpenGL 17 立方体贴图
立方体贴图 我们已经使用2D纹理很长时间了,但除此之外仍有更多的纹理类型等着我们探索。在本节中,我们将讨论的是将多个纹理组合起来映射到一张纹理上的一种纹理类型:立方体贴图(Cube Map)。 简单来说,立方体贴图就是一个包含了…...

【四 (6)数据可视化之 Grafana安装、页面介绍、图表配置】
目录 文章导航一、Grafana介绍[✨ 特性]二、安装和配置1、安装2、权限配置(账户/团队/用户)①用户管理②团队管理③账户管理④看板权限 3、首选项配置4、插件管理①数据源插件②图表插件③应用插件④插件安装方式一⑤安装方式二 三、数据源管理1、添加数…...
jvm 堆
Java虚拟机(JVM)中的堆是运行时数据区的一个主要部分,它用于存放对象实例和数组。它是所有Java线程共享的一块内存区域,是垃圾收集器管理的主要区域,因此也被称作垃圾收集堆(Garbage-Collected Heap&#x…...

Jenkins通知目标服务器拉取Harbor镜像部署
1.告诉目标服务器拉取哪个镜像 2.判断当前有没有正在运行此容器,有就删除 3.接着查看拉取的镜像目标服务器上是否已存在,有就删除 4.拉取Harbor镜像 5.运行容器 目标服务器编写脚本 创建个部署脚本 vim deploy.sh告诉目标服务器Harbor地址、仓库、镜像…...
Android 13.0 系统中framework中关于Activitity的生命周期的源码讲解
1.前言 在13.0的系统rom定制化开发中,在framework中对activitity的生命周期的掌握和了解也是非常重要的,这样有利于在启动某个app的activity的页面的时候,可以 监听到是在启动 resume stop的过程,也好进行相关的功能开发,接下来就分析下Activity的相关生命周期的代码 2.…...
常见的几个Python技术难题
大家在日常开发中有没有遇到一些难题呢?计划后面出几期专题针对性的解决。大家如果有其它问题可以在评论区给出哈。 以下是几个Python技术难题的例子: 并发和多线程编程:Python的全局解释器锁(GIL)限制了多线程的并行…...

【探索Linux】—— 强大的命令行工具 P.28(网络编程套接字 —— 简单的UDP网络程序模拟实现)
阅读导航 引言一、UDP协议二、UDP网络程序模拟实现1. 预备代码⭕makefile文件⭕打印日志文件⭕打开指定的终端设备文件,并将其作为标准错误输出的目标文件描述符 2. UDP 服务器端实现(UdpServer.hpp)3. UDP 客户端实现(main函数&a…...

【MATLAB源码-第165期】基于matlab的科莫多巨蜥算法(KMA)机器人栅格路径规划,输出做短路径图和适应度曲线。
操作环境: MATLAB 2022a 1、算法描述 科莫多巨蜥算法(Komodo Mlipir Algorithm,简称KMA)是一种受到印尼科莫多岛上独特生物——科莫多巨蜥启发的创新算法。尽管这个算法的名称听起来很有趣,但实际上它并不是一个公认…...

【Linux】项目部署CPU彪高如何定位
1.查看所有CPU占比 使用top指令获取彪高进程的PID 2.输出进程的信息 ps H -eo pid,tid,%cpu | grep 1313 3.查看线程的信息 jstack tid nid都是十六进制的 4.进制转换 将 tid的十进制转为十六进制 找到nid 可以定位到具体位置 5.关闭程序 ps -ef | grep java kill -9 jav…...

第十二届蓝桥杯大赛软件赛决赛C/C++ 研究生组-纯质数
直接判断数据过大 相对而言,由2,3,5,7组成的数更少,则先筛选出由2,3,5,7组成的数,再判断这些数中的质数个数即可 #include <iostream> using namespace std; int main() {printf("1903");…...
MyBatis面试简答题
以下是一份MyBatis的高难度简答题,共20题: 请解释MyBatis中#{}和${}的区别,并举例说明它们在实际应用中的使用场景。 MyBatis的Mapper接口是如何与XML映射文件关联的? 如何在MyBatis中实现动态SQL?请列举几种常见的动态SQL元素并解释其作用。 描述MyBatis中的ResultMap的作…...
lua 中的元表
a{ age0, __tostringfunction() { }, __callfunction() { }, } b{} a.__indexa{}//将a表中的__index指向自己 setmetatable(a,b)//将b设置为a的元表; __tostring 当子表a被当做字符串使用时会调用原表b中的__tostring方法, __call 当子表a被当做字符串使用时…...

c语言综合练习题
1.编写程序实现键盘输入一个学生的学分绩点 score(合法的范围为:1.0—5.0),根据学生的学分绩点判定该学 生的奖学金的等级,判定规则如下表所示。 #include <stdio.h>int main() {float score;printf("请输入学生的学分…...

相机拍照与摄影学基础
1.相机拍照 相机可能形状和大小不同,但基本功能相同,包括快门速度、光圈和感光度,这些是摄影的通用概念。即使是一次性相机也是基于这三个理念工作的。不同类型相机在这三个概念上的唯一区别是你可以控制这些功能的程度。这三个参数被称为相…...
Pytorch:torch.cuda.empty_cache()
torch.cuda.empty_cache() 原理 torch.cuda.empty_cache() 是PyTorch中用来释放未被分配的缓存的内存的函数。在使用GPU进行计算时,CUDA会在内部维护一个内存缓存池,以便更快地分配和释放。但有时候,这些缓存的内存在不再需要的时候&#x…...

Linux--gdb调试
一.安装gdb sudo apt install gdb 二.使用gdb 三.gdb的相关操作 gdb 可执行文件名 显示代码: l 加断点: b 行号 启动程序:r(运行之前一定要加断点) 查看断点信息: info break/info b 删除断点信息:delete 断点编号 单步执行:n 打印 :p 显示:display 变量名: 退出:q …...

wordpress后台更新后 前端没变化的解决方法
使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…...
[2025CVPR]DeepVideo-R1:基于难度感知回归GRPO的视频强化微调框架详解
突破视频大语言模型推理瓶颈,在多个视频基准上实现SOTA性能 一、核心问题与创新亮点 1.1 GRPO在视频任务中的两大挑战 安全措施依赖问题 GRPO使用min和clip函数限制策略更新幅度,导致: 梯度抑制:当新旧策略差异过大时梯度消失收敛困难:策略无法充分优化# 传统GRPO的梯…...
CVPR 2025 MIMO: 支持视觉指代和像素grounding 的医学视觉语言模型
CVPR 2025 | MIMO:支持视觉指代和像素对齐的医学视觉语言模型 论文信息 标题:MIMO: A medical vision language model with visual referring multimodal input and pixel grounding multimodal output作者:Yanyuan Chen, Dexuan Xu, Yu Hu…...

shell脚本--常见案例
1、自动备份文件或目录 2、批量重命名文件 3、查找并删除指定名称的文件: 4、批量删除文件 5、查找并替换文件内容 6、批量创建文件 7、创建文件夹并移动文件 8、在文件夹中查找文件...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...

c#开发AI模型对话
AI模型 前面已经介绍了一般AI模型本地部署,直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型,但是目前国内可能使用不多,至少实践例子很少看见。开发训练模型就不介绍了&am…...

有限自动机到正规文法转换器v1.0
1 项目简介 这是一个功能强大的有限自动机(Finite Automaton, FA)到正规文法(Regular Grammar)转换器,它配备了一个直观且完整的图形用户界面,使用户能够轻松地进行操作和观察。该程序基于编译原理中的经典…...

Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决
Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决 问题背景 在一个基于 Spring Cloud Gateway WebFlux 构建的微服务项目中,新增了一个本地验证码接口 /code,使用函数式路由(RouterFunction)和 Hutool 的 Circle…...

OPENCV形态学基础之二腐蚀
一.腐蚀的原理 (图1) 数学表达式:dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy) 腐蚀也是图像形态学的基本功能之一,腐蚀跟膨胀属于反向操作,膨胀是把图像图像变大,而腐蚀就是把图像变小。腐蚀后的图像变小变暗淡。 腐蚀…...