近期和几位做嵌入式开发的朋友闲聊过程中,一位朋友抱怨到:这C#太难用了,我想在N个窗体(或者是N个用户组件之间)传递值都搞不定,非得要定义一个全局变量来存储,然后用定时器来刷新值,太Low了。我急切的回答道:这很简单,不就是委托的事嘛。那你来一个示例啊:朋友道。此为这篇博客的起因,所以此篇博客对于有c#开发经验的伙伴们那是小菜一喋。

一、对委托的理解

委托:同一个功能,可以根据不同的场景委托给不同的方法具体执行; 举个栗子:某位美食爱好妹子,通常自己做美食;找到男票后,就男票做美食;换男票后,就第二任男票做美食。我们把这例子委托抽象化:

定义一个委托功能:做美食;规范及流程:输入”食材“,通过”做美食“的委托,输出”美食“。

委托实现之自己做:妹子自己做美食

委托实现之一号男票做:一号男票做美食

委托实现之二号男票做:二号男票做美食

做美食这项功能,被妹子在不同的时间段分配给了不同的对象,虽然妹子,男一,男二做的美食都很好吃,但味道肯定有区别。这就是委托生活化的示例,各位看观了解否(偷笑)。

二、代码实现

上面的示例如何用代码实现,这里就不展示了(真的很简单)。下面我们换一个稍有难度和实际应用的示例,需求说明:主窗体显示一个列表,子窗体增加数据(不关闭子窗体的情况下),主窗体列表自动更新,且第二个子窗体打开后,窗体内的列表也要同时更新。

UI设计:一个主窗体,两个子窗体(A窗体:增加数据,B窗体:显示数据),一个用户组件(列表显示内容)

2.1 EventBus实现

代码如下:

public class EventBus<T>
{
private List<T> list = new List<T>(); public event EventHandler<EventBusArg<List<T>>> EventNotice;
public delegate void DelegateItemInfoEvent(List<T> items); public void Add(T item)
{
this.list.Add(item);
this.TriggerEventNotice();
} public void Remove(T item)
{
this.list.Remove(item);
this.TriggerEventNotice();
} public List<T> GetAll()
{
return this.list;
} private void TriggerEventNotice()
{
if (this.EventNotice != null)
{
this.EventNotice.Invoke(this, new EventBusArg<List<T>>()
{
Data = this.GetAll()
});
}
}
} public class EventBusArg<T> : EventArgs
{
public T Data { get; set; }
}

重点:

1. 定义了一个委托类型:DelegateItemInfoEvent(List<T> items)

2. 定义了一个事件对象:EventHandler<EventBusArg<List<T>>>

3. 事件对象的参数必须继承EventArgs对象

4. 事件依赖委托

2.2 主窗体

代码如下:

        private EventBus<ItemInfo> eventBus = new EventBus<ItemInfo>();
private EventBus<ItemInfo>.DelegateItemInfoEvent FunItem;
public Form1()
{
InitializeComponent();
this.eventBus.EventNotice += EventBus_EventNotice;
} private void EventBus_EventNotice(object sender, EventBusArg<List<ItemInfo>> e)
{
if (this.ucList1.InvokeRequired)
{
FunItem = new EventBus<ItemInfo>.DelegateItemInfoEvent(RefreshItem);
this.ucList1.Invoke(FunItem, e.Data);
}
else
{
this.RefreshItem(e.Data);
}
} private void RefreshItem(List<ItemInfo> item)
{
var ls = this.eventBus.GetAll();
this.ucList1.LoadData(ls);
}

重点:

1. 捕获事件:this.eventBus.EventNotice += EventBus_EventNotice;

2. 事件处理方法中,需要判断是否为UI线程引发,如果不是,则需要委托来进行切换线程,代码见:private void EventBus_EventNotice(object sender, EventBusArg<List<ItemInfo>> e) 方法

3. 其中FunItem是委托类型的变量,其最终的实现为RefreshItem方法

2.3 A窗体:增加数据

代码如下:

private EventBus<ItemInfo> eventBus;
public Form2(EventBus<ItemInfo> eventBus)
{
this.eventBus = eventBus;
InitializeComponent();
} private void button1_Click(object sender, EventArgs e)
{
//在UI线程
this.eventBus.Add(new ItemInfo()
{
Title = textBox1.Text,
Val = Int32.Parse(textBox2.Text)
});
} private void button2_Click(object sender, EventArgs e)
{
//跨线程
Task.Factory.StartNew(() =>
{
for(var i=; i < ; i++)
{
this.eventBus.Add(new ItemInfo()
{
Title = i.ToString() + "-Title",
Val = i
});
}
});
}

重点:

1. 传入了EventBus对象的实例,此实例与主界面的EventBus实例为同一个【这点很重要,发布和订阅的事件必须在同一实例上】

2. button2_Click事件展示的是跨线程事件,执行此代码,主界面的刷新会走委托

2.4 B窗体:订阅列表显示

代码如下:

private EventBus<ItemInfo> eventBus;
public Form3(EventBus<ItemInfo> eventBus)
{
this.eventBus = eventBus;
InitializeComponent();
this.eventBus.EventNotice += EventBus_EventNotice;
} private void EventBus_EventNotice(object sender, EventBusArg<List<ItemInfo>> e)
{
if (this.ucList1.InvokeRequired)
{
var FunItem = new EventBus<ItemInfo>.DelegateItemInfoEvent(RefreshItem);
this.ucList1.Invoke(FunItem, e.Data);
}
else
{
this.RefreshItem(e.Data);
}
} private void RefreshItem(List<ItemInfo> item)
{
var ls = this.eventBus.GetAll();
this.ucList1.LoadData(ls);
} private void Form3_FormClosing(object sender, FormClosingEventArgs e)
{
this.eventBus.EventNotice -= EventBus_EventNotice;
}

重点:

1. 事件的订阅与取消订阅,一般情况下可以在关闭窗体时取消订阅

三、回顾

1. 事件依赖委托,事件可以订阅和取消订阅,其订阅就是为事件增加委托。

2. 委托的本质还是方法(或者说是函数),只不过方法变成了一个变量,可以在运行时动态改变

3. 源码下载

C#之委托如此简单的更多相关文章

  1. .net学习之多线程、线程死锁、线程通信 生产者消费者模式、委托的简单使用、GDI(图形设计接口)常用的方法

    1.多线程简单使用(1)进程是不执行代码的,执行代码的是线程,一个进程默认有一个线程(2)线程默认情况下都是前台线程,要所有的前台线程退出以后程序才会退出,进程里默认的线程我们叫做主线程或者叫做UI线 ...

  2. C#基础系列——委托实现简单设计模式

    前言:上一篇介绍了下多线程的相关知识:C#基础系列——多线程的常见用法详解,里面就提到了委托变量.这篇简单介绍下委托的使用.当然啦,园子里面很多介绍委托的文章都会说道:委托和事件的概念就像一道坎,过了 ...

  3. C#基础:对委托的简单理解

    在编程过程中,我们习惯把数据作为参数传递给方法(例:int a=int.Parse(“20”);).是否能将一个方法传递给另一个方法呢?是不是听起来有点奇怪!!! 线程大家应该熟悉吧,在计算机中并行运 ...

  4. C#委托的简单剖析

    为什么在Button1的Click事件发生之后,button1_Click方法就会被调用呢? 实际上,在我们双击Button1的时候,IDE自动的添加了一段代码,该段代码位于“Form1.Design ...

  5. C# 委托和事件(一):最简单的委托和事件

    C#的事件基于委托,所以先说委托. 一切脱离实际场景的抽象概念新手看上去就像是在扯犊子,不错,我就是个新手.所以我需要一个实际的场景. 明天刚好考试(商务英语),考试上有两个角色(class):老师( ...

  6. C# 委托简单使用方法

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

  7. [.NET] C# 知识回顾 - 委托 delegate (续)

    C# 知识回顾 - 委托 delegate (续) [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6046171.html 序 上篇<C# 知识回 ...

  8. .NET面试题系列[7] - 委托与事件

    委托和事件 委托在C#中具有无比重要的地位. C#中的委托可以说俯拾即是,从LINQ中的lambda表达式到(包括但不限于)winform,wpf中的各种事件都有着委托的身影.C#中如果没有了事件,那 ...

  9. [ASP.NET MVC 大牛之路]02 - C#高级知识点概要(1) - 委托和事件

    在ASP.NET MVC 小牛之路系列中,前面用了一篇文章提了一下C#的一些知识点.照此,ASP.NET MVC 大牛之路系列也先给大家普及一下C#.NET中的高级知识点.每个知识点不太会过于详细,但 ...

随机推荐

  1. JDK 1.8 之 Map.merge()

    Map 中ConcurrentHashMap是线程安全的,但不是所有操作都是,例如get()之后再put()就不是了,这时使用merge()确保没有更新会丢失. 因为Map.merge()意味着我们可 ...

  2. [Note] GNUstep on Windows

    1.下载与安装 www.gnustep.org/windows/installer.html 下载 GNUstep MSYS System GNUstep Core GNUstep Devel 并安装 ...

  3. Hadoop 学习笔记之 Incompatible namespaceIDs问题

    Hadoop重新格式化后,仍然datanode启动失败,查看log: org.apache.hadoop.hdfs.server.datanode.DataNode: Java.io.IOExcept ...

  4. MySQL系统表的利用姿势(浅探)

    MySQL数据库文件读写 权限要求: 具备读写权限并且目标文件为可读内容 目标内容具有完整路径且目录可访问 目标内容是否具备文件读写操作权限 查看是否有文件读写权限 show variables li ...

  5. 设置Activity全屏的方法:

    1)代码隐藏ActionBar 在Activity的onCreate方法中调用getActionBar.hide();即可 2)通过requestWindowFeature设置 requestWind ...

  6. springboot配置ehcache2.X缓存(@Cacheable等注解和手动操作缓存的工具类 支持element粒度的时间设置)

    本文只写出一些注意事项和源码,请善用官方文档~ 注解实现 @Cacheable @CachePut @CacheEvit 启动类上加@EnableCaching就可以开启缓存 由文档可知,自动检测缓存 ...

  7. 时间复杂度————被list.insert坑了

    今天被一个很简单的坑到了,还想了很长时间,insert 函数,真的知道它内部执行的操作吗? 开始其实是在看一本算法的书,书里面给了两段工作内容差不多的伪代码 第一段如下: data = [] whil ...

  8. shark恒破解笔记6-BC++假自效验

    这小节介绍了查壳(peid) 查软件编写语言(die)以及用esp定律脱aspack壳,最后是破解bc++的自校验部分 目标: 首先查看软件 peid查壳 有壳 ,但是不知道是什么语言写的,这里使用D ...

  9. muduo Library

    muduo是由陈硕(http://www.cnblogs.com/Solstice)开发的一个Linux多线程网络库,采用了很多新的Linux特性(例如eventfd.timerfd)和GCC内置函数 ...

  10. Rancher 2.1平台搭建及使用

    一.概述 1.1.什么是Rancher Rancher是一套容器管理平台,它可以帮助组织在生产环境中轻松快捷的部署和管理容器. Rancher可以轻松地管理各种环境的Kubernetes,满足IT需求 ...