【Unity技巧】统一管理回调函数——观察者模式
这次的内容有点类似设计模式里的观察者模式。但是和常规意义上的观察者模式也不是完全一致,所以各位就不要咬文嚼字啦!咦?设计模式?!不懂!没关系,说不定你以前就用过。
开场白
实现
using UnityEngine;
using System.Collections; public class TimerObserverOrSubject : MonoBehaviour { virtual protected void OnDestroy ()
{
if(Singleton.IsCreatedInstance("TimerController"))
{
(Singleton.getInstance("TimerController") as TimerController).ClearTimer(this);
}
}
}
TimerObserverOrSubject.cs的内容非常简单,它的工作就是在该脚本被析构时,及时地从计数器管理器里面删除涉及这个对象的所有Timer。
using UnityEngine;
using System.Collections;
using System.Collections.Generic; public class TimerController : MonoBehaviour { public delegate void OnCallBack(object arg);
public delegate bool OnIsCanDo(object arg); public class Timer {
public TimerObserverOrSubject m_Observer;
public OnCallBack m_Callback = null;
public object m_Arg = null; public TimerObserverOrSubject m_Subject;
public OnIsCanDo m_IsCanDoFunc = null;
public object m_ArgForIsCanDoFunc = null; public float m_PassTime = 0; public Timer(TimerObserverOrSubject observer, OnCallBack callback, object arg,
TimerObserverOrSubject subject, OnIsCanDo isCanDoFunc, object argForIsCanDo) {
m_Observer = observer;
m_Callback = callback;
m_Arg = arg; m_Subject = subject;
m_IsCanDoFunc = isCanDoFunc;
m_ArgForIsCanDoFunc = argForIsCanDo; m_PassTime = 0;
} public Timer(TimerObserverOrSubject observer, OnCallBack callback, object arg, float time) {
m_Observer = observer;
m_Callback = callback;
m_Arg = arg; m_Subject = null;
m_IsCanDoFunc = null;
m_ArgForIsCanDoFunc = null; m_PassTime = time;
}
}
private List<Timer> m_Timers = new List<Timer>();
private List<Timer> m_NeedRemoveTimer = new List<Timer>();
private List<Timer> m_CurRunTimer = new List<Timer>(); /// <summary>
/// Sets the timer.
/// </summary>
/// <param name='observer'>
/// The TimerObserverOrSubject you need to listen
/// </param>
/// <param name='callback'>
/// The callback when condition is true.
/// </param>
/// <param name='arg'>
/// Argument of the callback.
/// </param>
/// <param name='observer'>
/// The TimerObserverOrSubject you need to observe
/// </param>
/// <param name='isCanDoFunc'>
/// The condition function, must return a boolean.
/// </param>
/// <param name='argForIsCanDo'>
/// Argument for condition function.
/// </param>
public void SetTimer(TimerObserverOrSubject observer, OnCallBack callback ,object arg,
TimerObserverOrSubject subject, OnIsCanDo isCanDoFunc,object argForIsCanDo) {
if (observer == null || subject == null || callback == null || isCanDoFunc == null) return; if (isCanDoFunc(argForIsCanDo)) {
callback(arg);
return;
} Timer timer = new Timer(observer, callback, arg, subject, isCanDoFunc, argForIsCanDo);
m_Timers.Add(timer);
} /// <summary>
/// Sets the timer.
/// </summary>
/// <param name='observer'>
/// The TimerObserverOrSubject you need to listen
/// </param>
/// <param name='callback'>
/// The callback when time is up.
/// </param>
/// <param name='arg'>
/// Argument of the callback.
/// </param>
/// <param name='timepass'>
/// Timepass before calling the callback.
/// </param>
public void SetTimer(TimerObserverOrSubject observer, OnCallBack callback , object arg, float timepass) {
if (observer != null && callback != null) {
Timer timer = new Timer(observer, callback, arg, timepass);
m_Timers.Add(timer);
}
} /// <summary>
/// Clears all Timers of the observer.
/// </summary>
/// <param name='observer'>
/// The TimerObserverOrSubject you need to clear
/// </param>
public void ClearTimer(TimerObserverOrSubject observer) {
List<Timer> needRemovedTimers = new List<Timer>(); foreach (Timer timer in m_Timers) {
if (timer.m_Observer == observer || timer.m_Subject) {
needRemovedTimers.Add(timer);
}
} foreach (Timer timer in needRemovedTimers) {
m_Timers.Remove(timer);
}
} // Update is called once per frame
void Update ()
{
InitialCurTimerDict();
RunTimer();
RemoveTimer();
} private void InitialCurTimerDict() {
m_CurRunTimer.Clear(); foreach (Timer timer in m_Timers) {
m_CurRunTimer.Add(timer);
}
} private void RunTimer() {
m_NeedRemoveTimer.Clear(); foreach (Timer timer in m_CurRunTimer) {
if (timer.m_IsCanDoFunc == null) {
timer.m_PassTime = timer.m_PassTime - Time.deltaTime;
if (timer.m_PassTime < 0) {
timer.m_Callback(timer.m_Arg);
m_NeedRemoveTimer.Add(timer);
}
} else {
if (timer.m_IsCanDoFunc(timer.m_ArgForIsCanDoFunc)) {
timer.m_Callback(timer.m_Arg);
m_NeedRemoveTimer.Add(timer);
}
}
}
} private void RemoveTimer() {
foreach (Timer timer in m_NeedRemoveTimer) {
if (m_Timers.Contains(timer)) {
m_Timers.Remove(timer);
}
}
} }
首先,它定义了回调函数的类型:
public delegate void OnCallBack(object arg);
public delegate bool OnIsCanDo(object arg);
关于C#的委托机制,如果有童鞋不了解,请详见官方文档。简单来说,委托类似一个函数指针,常被用于回调函数。
/// <summary>
/// Sets the timer.
/// </summary>
/// <param name='observer'>
/// The observer to observe the subject
/// </param>
/// <param name='callback'>
/// The callback when condition is true.
/// </param>
/// <param name='arg'>
/// Argument of the callback.
/// </param>
/// <param name='subject'>
/// The subject you need to observe
/// </param>
/// <param name='isCanDoFunc'>
/// The condition function, must return a boolean.
/// </param>
/// <param name='argForIsCanDo'>
/// Argument for condition function.
/// </param>
public void SetTimer(TimerObserverOrSubject observer, OnCallBack callback ,object arg,
TimerObserverOrSubject subject, OnIsCanDo isCanDoFunc,object argForIsCanDo) {
if (observer == null || subject == null || callback == null || isCanDoFunc == null) return; if (isCanDoFunc(argForIsCanDo)) {
callback(arg);
return;
} Timer timer = new Timer(observer, callback, arg, subject, isCanDoFunc, argForIsCanDo);
m_Timers.Add(timer);
}
/// <summary>
/// Sets the timer.
/// </summary>
/// <param name='observer'>
/// The observer to observe the subject
/// </param>
/// <param name='callback'>
/// The callback when time is up.
/// </param>
/// <param name='arg'>
/// Argument of the callback.
/// </param>
/// <param name='timepass'>
/// Timepass before calling the callback.
/// </param>
public void SetTimer(TimerObserverOrSubject observer, OnCallBack callback , object arg, float timepass) {
if (observer != null && callback != null) {
Timer timer = new Timer(observer, callback, arg, timepass);
m_Timers.Add(timer);
}
}
它负责建立一个计数器,在timepass的时间后,通知observer,执行observer的callback(arg)函数。
例子
using UnityEngine;
using System.Collections; public class TimerSample : TimerObserverOrSubject { private TimerController m_TimerCtr = null; private bool m_IsCanDisplay = false; private string m_DisplayContent = "Hello, candycat!"; // Use this for initialization
void Start () {
m_TimerCtr = Singleton.getInstance("TimerController") as TimerController; //m_TimerCtr.SetTimer(this, Display, m_DisplayContent, 5); m_TimerCtr.SetTimer(this, Display, null, this, IsCanDisplay, null); StartCoroutine(DelayDisplay());
} void Display(object arg) {
if (arg == null) {
Debug.Log(m_DisplayContent);
} else {
string content = arg as string; Debug.Log(content);
}
} bool IsCanDisplay(object arg) {
return m_IsCanDisplay;
} IEnumerator DelayDisplay() {
yield return new WaitForSeconds(5.0f); m_IsCanDisplay = true;
} // Update is called once per frame
void Update () { }
}
首先,它向TimerController请求注册了一个计时器。这里,它的条件是IsCanDisplay函数,它返回bool值m_IsCanDisplay。而这个值将会在5秒后,通过协同函数DelayDisplay来由false置为true。当其为true时,TimerController就将通知TimerSample调用Display函数。
结束语
【Unity技巧】统一管理回调函数——观察者模式的更多相关文章
- Unity 之 C# 利用回调函数实现C++匿名函数
做C++开发的都用过匿名函数很好用,可是C#开发怎么实现呢?前几天做一个拍照功能的时候.我偶然发现某个函数假设是C++的话.用匿名函数太好了,于是開始研究C#的回调,代理.托付等,最后总算是实现了我想 ...
- C++回调函数(callback)的使用
什么是回调函数(callback) 模块A有一个函数foo,他向模块B传递foo的地址,然后在B里面发生某种事件(event)时,通过从A里面传递过来的foo的地址调用foo,通知A发生了什么事 ...
- [转]C++回调函数(callback)的使用
原文地址:http://blog.sina.com.cn/s/blog_6568e7880100p77y.html 什么是回调函数(callback) 模块A有一个函数foo,他向模块B传递fo ...
- jquery的2.0.3版本源码系列(6):2880-3042行,回调对象,对函数的统一管理
目录 1 . 回调对象callbacks的演示 回调的使用有一点像事件绑定,先绑定好,等到有点击事件或者其他时就触发. <script src="js/jquery-2.0.3.js& ...
- 前端基本知识(四):JS的异步模式:1、回调函数;2、事件监听;3、观察者模式;4、promise对象
JavaScript语言将任务的执行模式可以分成两种:同步(Synchronous)和异步(Asychronous). “同步模式”就是一个任务完成之后,后边跟着一个任务接着执行:程序的执行顺序和排列 ...
- 常用js,css文件统一加载方法,并在加载之后调用回调函数
原创内容,转载请注明出处! 为了方便资源管理和提升工作效率,常用的js和css文件的加载应该放在一个统一文件里面完成,也方便后续的资源维护.所以我用js写了以下方法,存放在“sourceControl ...
- Unity C# 调用 C++ DLL 并在 DLL 中调用 C# 的回调函数
Unity C# 调用 C++ DLL 并在 DLL 中调用 C# 的回调函数~~~ 呵呵... 看着有点晕.. 再解释一下就是 在Unity中 使用 C# 调用 C++ 写的 DLL, 但是在 ...
- 观察者模式在One Order回调函数中的应用
例如需求是搞清楚function module CRM_PRODUCT_I_A_CHANGE_ORGM_EC在什么样的场景下会被调用.当然最费时间的做法是设一个断点,然后跑对应的场景,观察断点何时被触 ...
- Lua中用table统一管理需要获取的unity物体
unity上的组件,可以用table统一管理 然后在初始化时候统一给table赋值,这样需要用到时候直接调用table中对应的key便可拿到对应的物体,省下了在脚本开头一堆声明的脚本,这样就不用声明这 ...
随机推荐
- solr和solrcloud
Solr = Lucene + Http(Servlet/REST) + Schema.xml+Solrconfig.xml Solr = SolrSingle + Solr MutilCore = ...
- Jedis分片Sentinel连接池实验
Jedis分片Sentinel连接池实验 1.起因 众所周知,Redis官方HA工具Sentinel已经问世很久了,但令人费解的是,Jedis官方却迟迟没有更新它的连接池.到目前Maven库中最新的2 ...
- Netty 4源码解析:请求处理
Netty 4源码解析:请求处理 通过之前<Netty 4源码解析:服务端启动>的分析,我们知道在最前端"扛压力"的是NioEventLoop.run()方法.我们指定 ...
- Sublime Text 3下C/C++开发环境搭建
Sublime Text 3下C/C++开发环境搭建 之前在Linux Mint 17一周使用体验中简单介绍过Sublime Text. 1.Sublime Text 3安装 Ubuntu.Linux ...
- 分析RunTime执行命令以及得到返回值
RunTime执行命令得到返回值 我们有在好好几篇博客里提到过RunTime,比如 JAVA之旅(二十三)--System,RunTime,Date,Calendar,Math的数学运算 Androi ...
- Appium移动自动化框架初探
作者:cryanimal QQ:164166060 本文简要介绍了appnium自动化框架的架构.加载流程.支持语言.相关配置,以及元素定位工具等. 官方网站: http://appium.io Ap ...
- EBS开发性能优化之SQL语句优化
(1)选择运算 尽可能先做选择运算,这是优化策略中最重要.最基本的一条,选择运算一般会使计算的中间结果大大变小,在对同一表格进行多个选择运算时,选择条件的排列顺序对性能也有很大影响,因为排列顺序不仅影 ...
- FORM内置系统函数
abort_query; 停止查询的执行 add_group_column(record grou ...
- 源码篇——Handler消息机制
Handler消息机制 Message 消息 Message.obtain() Message msg = new Message() Handler new Handler(){ handlerMe ...
- linux源码编译安装OpenCV
为了尽可能保证OpenCV的特性,使用OpenCV源码编译安装在linux上.先从安装其依赖项开始,以ubuntu 14.04.X为例讲解在Linux上源码编译安装OpenCV,其他linux版本可以 ...