由C#委托回调想到的二三事

news/2024/11/15 17:32:32/文章来源:https://www.cnblogs.com/geekfrank/p/18548348

写在前面:
之前的过开发程中,我愈发觉得面对复杂的界面要求,最好还是用UserControl将不同模块的界面设计单独封装,以应对客户频繁地需求更改。这样做能够在面对对不同的UI要求时,动态的加载预先设计好的特定模块的UserControl,不需要用代码对界面进行复杂的控制,否则要用代码控制一个个控件的生成与显示。设计之初费力,后面维护起来比较方便。
背景介绍:
最近开发新工具,针对不同的模块的数据展示我设计了不同的布局单独封装为UserControl,放置在PanelControl中作为数据展示。为了能够灵活的进行数据初始化,我给每个UserControl都订阅了主程序的通知事件。使用委托回调的方式,主程序中调用委托,子控件控件中自动执行具体的初始化方法。

//定义一个带有一个参数的无返回值的委托
delegate void DelegateUpdateUserControl(DeviceConfig item);
//声明委托对象
private DelegateUpdateUserControl UpdateUserControl;

大事件发生:
在测试过程中,数据的动态加载以及UserControl的切换展示没有问题。但随着切换次数的增加,界面变得越来越卡顿。
此时的我心中顿时警铃大作——有些该被释放掉的资源没有被释放掉?!但是每次进行界面切换,我都会调用Clear方法,将PanelControl里面的对象清空(PanelControl.Clear();),这时我开始怀疑UserControl对象仍在被主程序引用,系统无法进行GC.Collect(),对象没有被回收。
测试:
我在UserControl中的订阅方法里增加一行Console.WriteLine方法,进行测试。
理想情况下每切换一次UserControl,会调用一次委托,让UserControl绑定的委托方法输出一条信息。测试却发现,第一次会输出1条、第二次输出2条、第三次输出3条...
这说明了除了第一次调用委托方法,后面每次调用都有不止一个对象在响应且数量依次增加,看来确实是每次的UserControl都没有被成功回收。
结合.NET的垃圾回收机制,断定UserControl对象还在被主程序引用,导致没有被成功释放,可鞥是没有解除委托与方法的绑定造成的。
我将针对这个问题做了以下优化:
修改以下代码:

//在切换控件时,先清空原先的旧UserControl
//清空PanelControl所有控件,通常情况下只存在一个控件
PanelControlContainer.Controls.Clear();

改为:

if (PanelControlContainer.Controls.Count > 0)
{//获取到UserControl(PanelControlContainer)对象UControlProperty control =(UControlProperty)PanelControlContainer.Controls[0];//取消事件挂载,解除委托绑定UpdateUserControl -= control.UpdateControls;//从PanelControl中移除旧UserControlPanelControlContainer.Controls.Remove(control);//主动释放资源control.Dispose();
}

以上步骤主要是取消事件挂载,解除主程序与UserControl对象之间的引用关系,及时让GC回收资源。
结果:
经最后测试,调用委托后,只会有当前最新的UserControl对象响应,界面切换时卡顿的现象消失。

扩展——.NET垃圾回收机制:
遇到这种情况,我们必须对.NET的垃圾回收机制有一定了解,借此机会聊聊GC回收机制。以下内容参考《CLR via C#》(第四版)(CLR: Common Language Runtime,公共语言运行时):

  • 在C#中,当我们创建一个对象时,它的引用会被存储在栈(Stack)中,而对象的实际数据会被存储在堆(Heap)中。这是.NET运行时自动进行的内存管理,称为垃圾回收(Garbage Collection, GC)。
  • 每个对象都有两个开销字段:类型对象指针和同步块索引,在64位应用程序中各占8个字节。
  • 所有引用类型的变量都称为根对象,这些根对象是垃圾回收的起点,括局部变量、全局变量、静态字段等‌;
  • CLR开始回收时会暂停所有线程,,防止线程在CLR检查期间访问对象并更改其状态;
  • 然后CLR进入GC的标记阶段,CLR遍历堆中的所有对象,将同步索引块中的一位设为0;
  • 然后CLR检查所有活动根,任何根引用了堆上的对象,CLR都会标记那个对象,将同步索引块中的位设为1;
  • 一个对象被标记后,CLR会检查那个对象的根,标记他们引用的对象,如果对象已经被标记,就不重新检查对象的字段,避免引起死循环;
  • 检查结束后,已标记的对象至少有一个根在引用,我们说这种对象是可达的,不能被垃圾回收;
  • GC删除未被标记的对象,把被标记的对象挪到一块连续的空间,进行压缩,避免空间碎片化的问题;
  • 作为压缩的一个步骤,CLR还要从每个根减去所引用对象在内存中的偏移字节数,保证每个根引用的是之前的对象;
  • 最后恢复执行之前被暂停的线程;

总结:
给对象事件绑定委托,释放对象前,要主动解除委托绑定,避免出现资源无法释放的问题出现;
CLR是基于代的垃圾回收机制,自动工作并不定时释放不会再被访问的资源。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

虚拟串口工具和串口调试工具详解 - 附下载地址

简介 串口开发过程中, 一般需要以下工具用于开发和调试:虚拟串口工具简介 虚拟串口软件, 可以在系统中虚拟出串口, 这样开发人员可以在没有物理串口设备的情况下进行开发. 串口调试工具简介 串口调试工具主要用于给串口发送信息, 测试串口是否连通, 发送消息是否正常被接收等.本…

怎么用云游戏玩Steam?ToDesk云电脑新手入门教程

对于新手玩家来说,想要上手Steam游戏,这门槛真有点高。不说要从众多真假难辨的软件中找出正版,遇到Steam内想玩的游戏还得等着下载安装解压,费时又费力。玩家想解决这个困难也很简单,只需下个ToDesk云游戏即可。它是ToDesk云电脑中专门玩游戏的版块,预安装了上百款热门游…

Vuex与Redux比较

由于Vuex和Redux都是从Flux中衍生出来,同时Vuex对Redux部分思想也有一些借鉴,所以Vuex和Redux有很多相同点。很多资料也有介绍两者的对比,但大部分讲解的比较抽象,较难理解。笔者整理两者异同点,同时配有标准案例进行说明。注意本文不是科普vuex和redux相关概念,相关知识…

第6篇 Scrum 冲刺博客

作业要求这个作业属于哪个课程 计科34班这个作业的要求在哪里 团队作业4——项目冲刺这个作业的目标 1.站立式会议2.发布项目燃尽图3.每人的代码/文档签入记录4.适当的项目程序/模块的最新(运行)截图5.每日每人总结会议照片昨日已完成的工作/今天计划完成的工作成员 昨天已完…

RabbitMQ 五种模式

RabbitMQ是一种常用的消息队列服务,它提供了五种消息模型:简单模型、工作队列模型、发布/订阅模型、路由模型、主题模型。1. 简单模型(Simple Message Queue,简称SQS):一个生产者,一个消费者,一个队列。 2. 工作队列模型(Work Queue):多个消费者共同处理一个队列中的…

如何防止手机被远程控制,安全远控推荐ToDesk

随着电子设备及各样应用的兴起,手机可以为人们带来的便利已越来越多,从二十年前的联络通话,到现如今的社交娱乐、导航、缴费等;通过智能手机中的软件均可轻松实现。 然而虽然手机的妙用有很多,但对于一些不太善用电子设备的中老年亲友来说,在使用中却也存在一定的被诈骗风…

Docker不再神秘 ------Ubuntu20.04 安装Docker 及实用技巧,建议收藏

Dockerdocker是一种容器,简而言之就是别人把一堆环境配置好了,你可以下载下来直接拿来使用(我的个人理解),有点像虚拟机你知道吧。比如下面这样,我直接打开了一个小电脑(docker),里面桌面啊、root啊全都有,跟你ubuntu系统类似,单说细节还不完全一样,毕竟它轻便哈哈…

SELF-REFINE: Iterative Refinement with Self-Feedback

1. 概述 基于给定的Prompt,大语言模型生成的Reponse可能不是最好的(这一点我认为当前的LLM大部分都是Decoder架构,基于已生成的结果产生下一个Token,一旦之前生成结果出错,也不容易及时改正。)。 本文为原始的生成添加了额外的反思重写步骤,过程如下:对于给定的\(Input…

基于stm32的bacnet协议

bacnet协议对于国内网站来说,几乎可以说资料为零,通俗大论一遍,具体操作方法屁都没说 先从工具说起 开发工具 BACnetScan:(讯绕提供)(工具1) 链接:https://pan.baidu.com/s/1TJxc0xaEsCT3lJOlG78B7w 提取码:t7bw Yabe:(工具2) 链接:https://pan.baidu.com/s/1jfs…

未能加载文件或程序集 “项目名称对应的程序集,Version=1.0.0.0.culture=neutral.PublicKeyToken=null或它的某一个依赖项。系统找不到指定的文件。

Visual Studio 2022, AutoCAD开发, wpf项目, 因viewmodel中代码出现问题, 造成窗体设计器中无法预览(这个问题通过修改viewmodel代码解决), 删除项目路径下的obj及bin文件夹后, 重新生成项目, 出现新的错误:窗体能够显示了, 但个别控件无法正常显示, 以为是visual …

.Net Core关于项目引用和命名空间导入的一个小坑

.Net Core关于项目引用和命名空间导入的一个小坑 一、.Net Core项目嵌套引用的情况 经笔者测试验证,发现对于一个.Net Core项目Root,其引用另一个.Net Core项目Root.SubA后,Root项目会自动将Root.SubA项目引用的子项目,也纳入其引用项目池中,而无需再手动引用这些子项目。…

考研打卡(18)

开局(18) 开始时间 2024-11-15 16:36:35 结束时间 2024-11-15 16:53:39等会去打剧本杀数据结构1 以下________排序算法的最坏时间复杂度可以做到O(nlog(n))(暨南大学 2010年) A 归并排序 B 快速排序 C 冒泡排序 D 插入排序A 答案归并排序:基于分治的思想,其时间…