刨根问底U3D---关于Delegate的方方面面

我是否需要阅读这篇文章

Code1:

1 private delegate void BloodNumDelegate ();
2
3 public delegate void ExpNumChangeDelegate (int _currentExpNum);
4
5 public event ExpNumChangeDelegate mOnExpNumChangeDelegate;

Code2:

PlayerVoProxy.instance.mOnExpNumChangeDelegate += delegate(int _expNum)
{
    Console.WriteLine ("阿呆收到, ExpNum is : {0}", _expNum);
};

如果你可以很清楚明白上面两部分代码是什么意思,说明你已经很了解Delegate(至少比我了解:) ),

所以我想下面的文章也许对你的帮助不是很大. 不过如果某些地方你也有所疑惑,希望你能从我下面的文章中得到一些帮助.


这篇文章都包含哪些内容

这篇文章从最基本的Delegate开始介绍起,后续会引申到event, 匿名Delegate 及我个人对其使用的一点理解

提前需要阅读的文章

如果你像我当初一样对Delegate完全不理解,请先阅读

NET之美:.NET关键技术深入解析.NET之美:.NET关键技术深入解析 迷你书 中的 第3章 (C# 中的委托和事件).  

(作者对Delegate理解的及其深入,我是比不上了 :( )


什么是 Delegate

从观察者模式说起

观察者模式大家肯定都不陌生

using System;
using System.Collections;

namespace L1
{
    public class FakeBloodChangeDelegate
    {
        public ArrayList mAllFakeDelegateListener;

        public FakeBloodChangeDelegate ()
        {
            mAllFakeDelegateListener = new ArrayList ();
        }

        public void addListener (IFakeBloodNumChangeDelegateListener _listener)
        {
            mAllFakeDelegateListener.Add (_listener);
        }

        public void upldateListener ()
        {
            foreach (IFakeBloodNumChangeDelegateListener listener in mAllFakeDelegateListener)
            {
                listener.onBloodNumChange ();
            }
        }
    }
}


using System;

namespace L1
{
    public interface IFakeBloodNumChangeDelegateListener
    {
        void onBloodNumChange ();
    }
}


using System;

namespace L1
{
    public class Idiot : IFakeBloodNumChangeDelegateListener
    {
        public Idiot ()
        {
        }

        public void onBloodNumChange ()
        {
            Console.WriteLine ("Idiot --> onBloodNumChange");
        }
    }
}
using System;

namespace L1
{
    public class TestArea1
    {
        private FakeBloodChangeDelegate mFakeBloodChangeDelegate;
        private Idiot mIdiot;

        public TestArea1 ()
        {
        }

        public void run ()
        {
            mFakeBloodChangeDelegate = new FakeBloodChangeDelegate ();
            mIdiot = new Idiot ();

            mFakeBloodChangeDelegate.addListener (mIdiot);

            mFakeBloodChangeDelegate.upldateListener ();
        }
    }
}

这是一个比较标准的观察者, idiot是监听的人,当主体(FakeBloodChangeDelegate)发生变化时候调用 updateListener方法告诉所有监听对象

Ok, 接下来让我们稍微对这个FakeBloodChangeDelegate类进行一个简单的封装, 为其创建一个Player对象,Blood是Player的一部分

using System;

namespace L1
{
    public class FakePlayerVoProxy
    {
        public FakeBloodChangeDelegate bloodNumChangeListner;

        private int mCurrentBloodNum = 0;

        public FakePlayerVoProxy ()
        {
            bloodNumChangeListner = new FakeBloodChangeDelegate ();
        }

        public void addPlayerBlood (int _addNum)
        {
            mCurrentBloodNum += _addNum;
            bloodNumChangeListner.upldateListener ();
        }

    }
}

于是,主类中的调用及变为:

using System;

namespace L1
{
    public class TestArea1
    {
        private FakePlayerVoProxy mPlayerVo;
        private Idiot mIdiot;

        public TestArea1 ()
        {
        }

        public void run ()
        {
            mPlayerVo = new FakePlayerVoProxy ();
            mIdiot = new Idiot ();

            mPlayerVo.bloodNumChangeListner.addListener (mIdiot);
            mPlayerVo.addPlayerBlood (10);
        }
    }
}

运行一下代码,会得到和上面一样的输出。 毕竟其实只是简单的对FakeBloodChangeDelegate进行一个封装,没有改变任何的代码。

如果上面的例子可以完全理解,就让我们往下继续

Delegate 就是一个类

这个其实是一个很重要的信息,在上面提到的那本迷你书里面已经说的非常清楚了. 我之所以把代码蹩脚的写成这个模样 其实也是为了说明这件事。

上面例子中的FakeBloodChangeDelegate,包括对应的Listener可以简化为一行代码

public delegate void BloodNumChangeDelegate ();

先抛开public不说, 后面用 delegate void BloodNumChangeDelegate(); 相当于就是告诉编辑器为你创建出 一个

FakeBloodChangeDelegate 和 IFakeBloodNumChangeDelegateListener (其实这块是不准确的表达,不过目前可以先不要细纠结到时是怎么回事,当看完后面的内容后 可以回头再去看 上面我提到的那本迷你书,里面讲的很详细)

using System;

namespace L1
{
    public class FakePlayerVoProxy
    {
        public delegate void BloodNumChangeDelegate ();

        public BloodNumChangeDelegate bloodNumChangeListner;

        private int mCurrentBloodNum = 0;

        public FakePlayerVoProxy ()
        {
        }

        public void addPlayerBlood (int _addNum)
        {
            mCurrentBloodNum += _addNum;
            bloodNumChangeListner ();
        }
    }
}

注意

public delegate void BloodNumChangeDelegate (); 这行表示创建了FakeBloodChangeDelegate 类其实和 FackPlayerVoProxy这里类没有任何关系,

完全可以单独建立一个文件,起名Apple,然后把这行放在那个Apple的类里面 不过根据这个类(Delegate)创建对象时候你就不能写

public BloodNumChangeDelegate bloodNumChangeListner;

而要写

public Apple.BloodNumChangeDelegate bloodNumChangeListner;

个人关于Delegate的一点理解

因为是写AS出身(ActionSprite),所以对于回调函数一点都不陌生,而Delegate就可以理解为C#中的回调函数,唯一会比较费解的会是

public delegate void BloodNumChangeDelegate ();
public BloodNumChangeDelegate bloodNumChangeListner;

这两行代码,这块的核心就是要理解 Delegate其实就是告诉编译器为你生成一个实现了观察者模式的类.



关于Delegate前面的Public修饰符

之前我在解释Delegate是怎么一回事时候,故意忽略掉了其前面的类修饰符 public. 这块其实还是很有文章的. 对于Delegate或者回归本质,对于观察者模式,最主要的三个函数应该是

addListener,  removeListener,  updateListener

而OO的思想就是 一定要封装 对于外部监听者 只应该能访问到 addListener, removeListener   而upldateListner 这个应该只有主类才可以做的事情.

让我们先退回到实现Delegate之前的例子上面,继续使用FakeBloodChangeDelegate ,

若要实现以上的需求,仅需要对应的将FakeBloodChangeDelegate改为private 并提供相应的public函数即可

using System;

namespace L1
{
    public class FakePlayerVoProxy
    {
        private FakeBloodChangeDelegate mBloodNumChangeListner;

        private int mCurrentBloodNum = 0;

        public FakePlayerVoProxy ()
        {
            mBloodNumChangeListner = new FakeBloodChangeDelegate ();
        }

        public void addListener(IFakeBloodNumChangeDelegateListener _listener)
        {
            mBloodNumChangeListner.Add(_listener);
        }

        public void removeListener(IFakeBloodNumChangeDelegateListener _listener){//对应Remove代码}

                 public void addPlayerBlood (int _addNum)
        {
            mCurrentBloodNum += _addNum;
            bloodNumChangeListner.upldateListener ();
        }

    }
}

这块应该很好理解吧?  所以呢如果使用Delegate去实现同样的操作及可以写成:

using System;

namespace L1
{
    public class FakePlayerVoProxy
    {
        public delegate void BloodNumChangeDelegate ();

        private BloodNumChangeDelegate bloodNumChangeListner;

        private int mCurrentBloodNum = 0;

        public FakePlayerVoProxy ()
        {
        }

        public void addListener (?? _listener)
        {
            bloodNumChangeListner += _listener;
        }

        public void addPlayerBlood (int _addNum)
        {
            mCurrentBloodNum += _addNum;
            bloodNumChangeListner ();
        }

    }
}

你会发现 在addListener这里 不知道该怎么写了(其实是可以写的,不过如果这块你都理解怎么写了也就不用再继续往下读啦 哈),  这个 _listener应该是什么类型呢?



使用Event关键字

在推荐的那本迷你书里面(什么?! 还没看...) 已经把这块说的非常明白了, 为了封装Delegate,对外仅提供add,remove 但是不提供

update方法, 但是如果不写成public 又没办法对外访问,所以及产生了 Event这个关键字

using System;

namespace L1
{
    public class FakePlayerVoProxy
    {
        public delegate void BloodNumChangeDelegate ();

        public event BloodNumChangeDelegate bloodNumChangeListner;

        private int mCurrentBloodNum = 0;

        public FakePlayerVoProxy ()
        {
        }

        public void addPlayerBlood (int _addNum)
        {
            mCurrentBloodNum += _addNum;
            bloodNumChangeListner ();
        }

    }
}
using System;

namespace L1
{
    public class TestArea1
    {
        private FakePlayerVoProxy mPlayerVo;
        private Idiot mIdiot;

        public TestArea1 ()
        {
        }

        public void run ()
        {
            mPlayerVo = new FakePlayerVoProxy ();
            mIdiot = new Idiot ();

            mPlayerVo.bloodNumChangeListner += mIdiot.onBloodNumChange;

            mPlayerVo.addPlayerBlood (10);
        }

    }
}

如果在TestArea1中调用 mPlayerVo.bloodNumChangeListner ();

会收到

Error(14,14): Error CS0070: The event `L1.FakePlayerVoProxy.bloodNumChangeListner‘ can only appear on the left hand side of += or -= when used outside of the type `L1.FakePlayerVoProxy‘ (CS0070) (L1)

也就是说在类的外部是无法调用"updateListener" 这个方法了. 这样也就符合了OO得封装原则



结束语

到此, Delegate和Event的用法 我想大家应该有了一个初步的认识,后续只要在项目中多试几次 应该就能有很好的掌握吧

(其实我接触C#也才不到一周,所以能理解的也就到这里了,希望对大家有帮助.)



等等! Code2中的 代码是什么?

PlayerVoProxy.instance.mOnExpNumChangeDelegate += delegate(int _expNum)
{
    Console.WriteLine ("阿呆收到, ExpNum is : {0}", _expNum);
};

其实这个代码 可以理解为匿名函数( 其实是匿名委托)

比如对于一个按钮

private void fun1()
{
   dummyBtn.onClick+=fun2
}

private void fun2()
{
   //Fun2Code
}

但是如果fun2的代码很简单 或者其他原因 也可以使用 匿名委托 也就是如上的写法

private void fun1()
{
   dummyBtn.onClick+=delegate()
   {
       //Fun2Code
   }
}

一般在按钮的处理以及Tween操作时候大多愿意这样去做,不过我个人还是不太爱使用匿名委托(As3中是匿名函数)这种方式做处理. 因人而异吧.



真的不能使用 private BloodNumChangeDelegate bloodNumChangeListner 么?

using System;

namespace L1
{
    public class FakePlayerVoProxy
    {
        public delegate void BloodNumChangeDelegate ();

        private BloodNumChangeDelegate bloodNumChangeListner;

        private int mCurrentBloodNum = 0;

        public FakePlayerVoProxy ()
        {
        }

        public void addListener(BloodNumChangeDelegate _listener)
        {
            bloodNumChangeListner += _listener;
        }

        public void addPlayerBlood (int _addNum)
        {
            mCurrentBloodNum += _addNum;
            bloodNumChangeListner ();
        }

    }
}
using System;

namespace L1
{
    public class TestArea1
    {
        private FakePlayerVoProxy mPlayerVo;
        private Idiot mIdiot;

        public TestArea1 ()
        {
        }

        public void run ()
        {
            mPlayerVo = new FakePlayerVoProxy ();
            mIdiot = new Idiot ();

            mPlayerVo.addListener (mIdiot.onBloodNumChange);

            mPlayerVo.addPlayerBlood (10);
        }

    }
}

以上两段代码是可以编译过并且可以运行的,  这个是我突然看到匿名委托使用的是Delegate修饰符所想到的,不过我对这块理解的也不是很透  如果有人知道原因 麻烦也给我解释一下吧

相关的内容 也许和 温故而知新:Delegate,Action,Func,匿名方法,匿名委托,事件 这篇文章提到的内容有关.

That‘s All

Eran.

时间: 2024-08-27 21:21:37

刨根问底U3D---关于Delegate的方方面面的相关文章

在Unity中使用事件/委托机制(event/delegate)进行GameObject之

欢迎来到unity学习.unity培训.unity企业培训教育专区,这里有很多U3D资源.U3D培训视频.U3D教程.U3D常见问题.U3D项目源码,[狗刨学习网]unity极致学院,致力于打造业内unity3d培训.学习第一品牌. 一对多的观察者模式机制有什么缺点? 如果你对如何在Unity中使用事件/委托机制还不太了解,建议您查看我的前一篇文章:[Unity3D技巧]在Unity中使用事件/委托机制(event/delegate)进行GameObject之间的通信 在前一篇博客里面,我们写到

刨根问底U3D---如何退出Play模式后保留数据更改

实际中遇到的需求 在做一款对抗类游戏,目前正在调整游戏的平衡性 所以就产生了一个需求 希望可以在Play模式时候对数据源做的更改可以在退出时候被保存下来. 举个Case, 比如 有一个炮塔 可以发射子弹, 然后有一组敌人 去攻击这个炮塔. 首先点击Play按钮 开始执行游戏逻辑 接着在Inspector中调整相应的数值,比如 射速啊,血量啊,敌人移动速度等等. 按原始的做法 就是调整以后 把所有的数据记录下来(截图或者写纸上) 点Stop进入编辑模式,所有数据回滚原始的 然后再一个一个把刚才改过

u3d局域网游戏网络(c# socket select 模型)

之前写了一篇. 发完之后第二天实际应用到游戏之后还是发现了一些小毛病. 比如网络模块有重复使用(多对象)的情况.所以将静态类该成了普通类. 比如安卓下会有些异常出现导致游戏逻辑不正常.所以网络相关的函数有些加了try块. 然后发现写入固定ip的方式根本不适合局域网.于是加了udp做的广播系统,用以服务器和客户端查找ip. udp广播部分和tcp不一样.因为没有连接,所以socket不需要shutdown.我在这里吃了一亏才知道. 别的没什么修改.贴上修正和扩展之后的代码. 有缘之人自取.唯一要求

c# socket select 模型代码(u3d)

其实写过多次网络链接.但是因为换了工作,又没电脑在身边,所以以前的代码都没办法翻出来用. 所以从今天起,一些常用的代码只好放到网上. 公司有一个局域网的游戏.本来想用u3d的rpc就可以完成.但是后来说要传语音.于是只要写一个tcp. 先给出服务器和客户端的socket代码段.其他依赖的一些文件,就不贴了. 目前完成的模块大的说就两块,网络和消息分发. 网络部分服务器有玩家池的管理. 网络部分没有想得很详细.因为是局域网,所以也不存在多大开销.如果有需要上千的需求,可能还要优化下代码结构以及锁.

在Unity中使用事件/委托机制(event/delegate)进行GameObject之间的通信

欢迎来到unity学习.unity培训.unity企业培训教育专区,这里有很多U3D资源.U3D培训视频.U3D教程.U3D常见问题.U3D项目源码,[狗刨学习网]unity极致学院,致力于打造业内unity3d培训.学习第一品牌. 引子 在前面两篇文章: 我们了解了2D中的Sprite,Animation,RigidBody和Collider,在继续开发游戏的过程中,我们会遇到这样的问题,如何处理GameObject之间的相互调用,比如说在FlappyBird中我们在小鸟撞倒管子的时候,要把这

U3D C# 实现AS3事件机制

写了很多年的AS3,最近接触U3D感觉事件机制没AS3的爽.咬紧牙关一鼓作气 基于C# 的委托实现了一版.废话不多说上干货. EventDispatcher代码如下: using UnityEngine;using System.Collections;using System;using System.Collections.Generic;/** * EventDispatcher 类是可调度事件的所有类的基类 * @author 回眸笑苍生**/public abstract class

温故而知新:Delegate,Action,Func,匿名方法,匿名委托,事件

一.委托Delegate 一般的方法(Method)中,我们的参数总是string,int,DateTime...这些基本的数据类型(或者没有参数),比如 public void HelloWorld() { Console.WriteLine("Hello World!"); } public void HelloWorld(string name) { Console.WriteLine("Hello ,{0}!", name); } 但是有些时候,我们希望把一

U3D面试五

U3D面试题 配置Unity3D调试环境 Visual Studio Tools for Unity 访问http://unityvs.com 安装对应的版本 使用方法(生成项目文件,如何调试) Array(数组)和List<T>的异同点? 相同点: 存储一组类型相同的数据 都可以通过"[i]"数组访问符获取内部数据 不同点: Array声明时需要指定固定的大小,且以后再也无法改变大小:List<T>声明时无需设定大小,且以后可以Add和Remove元素. Un

代理Delegate的小应用(代理日期控件和下拉框)

前言 在平时关于表格一类的的控件使用中,不可避免需要修改每个Item的值,通过在Item中嵌入不同的控件对编辑的内容进行限定,然而在表格的Item中插入的控件始终显示,当表格中item项很多的时候,会影响表格的美观和用户的体验.此时Delegate代理就派上了用场,通过Delegate可以使得Item的控件在编辑状态才显示,提高了用户的体验满意度. 效果展示 1.展示状态: 2.编辑状态   设计思路 这类效果的实现主要使用了QItemDelegate类,QItemDelegate类为数据项It