专栏首页喵叔's 专栏使用null条件运算符调用事件处理程序

使用null条件运算符调用事件处理程序

对于刚接触事件处理的开发人员来说,会觉得触发事件是一个非常容易的事情,只需要把事件定义好在触发的时候调用相关事件就可以了。但是实际上触发事件不是那么的简单,我们在这里考虑两个问题:

  1. 如果在程序中根本没有任何一个处理程序和某个事件关联,会出现什么情况?
  2. 如果存在多个线程都要检测并调用同一个事件,这些线程之间又存在争夺的问题,会出现什么情况? 针对上面这两个问题,在 C# 6.0 中新增的 null 条件运算符就可以解决这个问题。下面我们先来看一下简单的代码段。
//不安全的方式
public class EnventSource
{
    private int count;
    private EventHandler<int> Updated;
    public void RaiseUpdates()
    {
        count++;
        Updated(this,count);
    }
}

上面的代码中存在一个问题,如果对象触发 Updated事件时并没有相关的事件处理程序和它关联,这时就会出现 NullReferenceException 问题,在 C#6.0 出来之前如果要解决这个问题我们需要在每次触发前都要去判断以下事件处理程序是否为 null:

//C#6.0以前的处理方式
public class EnventSource
{
    private int count;
    private EventHandler<int> Updated;
    public void RaiseUpdates()
    {
        count++;
        if(Updated!=null){
            Updated(this,count);
        }
    }
}

经过修正后的代码可以在绝大部分情况下解决前面所提到的问题。注意我这里说的时绝大部分情况,还有一种特殊的情况会出现前面所提的问题,比如 A 线程在执行完 if 语句后发现 Updated 并不等于空,这时在 A 线程还没开始执行 Updated(this,count) 语句时 B 线程将事件处理程序的订阅解除了,那么在 A 线程执行到 Updated(this,count) 语句时事件处理程序已经为 null 了,这样仍然会出现 NullReferenceException 问题。针对这个问题开发人员会进行如下的处理:

//C#6.0以前的处理方式进一步修改
public class EnventSource
{
    private int count;
    private EventHandler<int> Updated;
    public void RaiseUpdates()
    {
        count++;
        var handler=Updated;
        if(handler!=null){
            handler(this,count);
        }
    }
}

上面的代码完美的处理的前面所说的问题,但是这样的代码会造成不易理解,我为什么修改成这样就是线程安全的呢?这是因为我们把事件处理程序赋值给了一个新的局部变量,这个局部变量就包含了多播委托,这个委托就可以应用原来的那个委托的所有成员变量里的事件处理程序。这种方法叫做浅拷贝,也就是创建了一个新的引用并让它指向了原来的事件处理程序。当一个线程把事件处理程序注销掉时,它只是修改的类实例中 Updated 子字段,而不是把处理程序从 handler 中移除掉。简单地说 handler 其实时 Updated 的快照,在触发事件的时候它所通知的那些事件处理程序其实是在做快照时记录下来的。这种方法虽然写法没错,但是对于新手来说是很难理解的,并且只要是在有触发事件的地方都要重复编写一边这样的代码。在 C#6.0 以后我们就可以使用 null 条件运算符来简单的处理这个问题,下面我们来看一下在 C#6.0 中如何解决这个问题。

public class EnventSource
{
    private int count;
    private EventHandler<int> Updated;
    public void RaiseUpdates()
    {
        count++;
        Updated?.Invoke(this.count);
    }
}

这段代码采用了 null 条件运算符安全的调用了事件处理程序,它首先会判断 ? 号左侧内容是否为 null,如果不为 null 则执行右侧的内容,反之跳过该语句执行下一条语句。这种方式的优势在于和以前使用 if 的方式相比,运算符左侧的内容只会计算一次。但是这里又有需要注意的地方,因为 C# 不允许在 ?. 后面出现括号,因此我们必须使用 Invoke 方法去触发事件,每定义一个委托或者事件编译器就会生成类型安全的 Invoke 方案,这就表明通过调用 Invoke 方法触发事件和以前的写法是完全相同的。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 第三章--第一节:条件判断语句

    从这一节开始,我们就进入到了进阶的阶段,这一章是在前一章的基础上进行提高扩展的,从本章开始,我会在每一节的结尾留下作业,大家可以将作业提交到我的邮箱或者直接将作...

    喵叔
  • 正确调用事件处理程序

    不管是刚接触 C# 还是已经具有多年开发经验的大部分人会觉得事件处理很简单,只需要把事件定义好然后在需要的时候出发它就可以了。其实这种想法是错误的,这里面有很多...

    喵叔
  • FormattableString 取代特定区域字符串

    有些软件系统是针对全球来开发的,因此一些字符串需要根据不同地区不同语言做出特定的处理。如果针对不同地区不同用语言分别编写字符串处理方法的话代码量是巨大的。那么这...

    喵叔
  • 获取ztree树的选中子菜单信息并且提交给后端

    前面写过,ztree实现一棵树的文章,https://www.jianshu.com/p/c2b919e91e91 现在要用ajax+json模拟交互效果

    王小婷
  • 使用Optional来减少null检查

    平常我们使用null检查在项目中简直太常见了,从数据库中查询到的数据可能不存在返回null,service中处理中发现不存在返回一个null,在互相调用的时候每...

    Dylan Liu
  • js中数据类型的相关问题

    李才哥
  • 如何使用Java调用CM的API动态配置Yarn资源池

    用户在使用CDH集群大数据平台时会有需求在自己的统一管理平台上通过API接口能够动态的设置Yarn资源池,Cloudera Manager提供了丰富的API接口...

    Fayson
  • [代码优化]null校验的优美处理

    我们写java代码的时候,使用对象前,都会下意识先判断对象非null,这是防止NPE的无奈之举,毕竟入门写代码时都写过npe的代码。这么做真的好吗,每层方法中都...

    逝兮诚
  • Tomcat中BIO与NIO

    Connector是一个桥梁它把Server和Engine链接了起来,Connector的作用是接受客户端端的请求,然后把请求委托为engine容器去处理。 ...

    加多
  • 解析Android 8.1平台SystemUI 导航栏加载流程

    需求开始做之前,一定要研读SystemUI Navigation模块的代码流程!!!不要直接去网上copy别人改的需求代码,盲改的话很容易出现问题,然而无从解决...

    砸漏

扫码关注云+社区

领取腾讯云代金券