介绍

先举一个简单的例子:

在UGUI中新建一个Button和Text,要求实现点击Button改变Text中的文字。

我的第一反应就是在Button上添加一个脚本,获取点击事件来改变Text的内容。

或者在Text和Button上都添加一个脚本,Text设置为单例模式,Button点击事件触发Text中的改变文字方法。

可是这两种方法要不就是关联度太高协作开发时很不好,要不在大项目中用很多单例模式很不好。

这时就需要到了解耦合技术!!!


高度解耦合就是利用事件的监听、事件的广播、委托来实现的。

无参实现

例如上面点击按钮切换的实现。

先写3个需要之后调用的类。

EventType

//这里面放的是一个枚举类型的事件码,就是用来表示按钮点击事件触发的函数
public enum EventType
{
ShowText,
}

CallBack

//这里面放的是所有不同参数类型的委托
public delegate void CallBack();
public delegate void CallBack<T>(T arg);

EventCenter

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine; public class EventCenter
{
//定义一个字典存放 事件码 和 委托
private static Dictionary<EventType, Delegate> m_EventTable = new Dictionary<EventType, Delegate>(); //无参的添加事件监听
public static void AddListener(EventType eventType,CallBack callBack)
{
//先判断是否包含
if (!m_EventTable.ContainsKey(eventType))
{
m_EventTable.Add(eventType, null);
}
//判断要添加的委托和已有的类型是否相同,只有相同才可以添加
Delegate d = m_EventTable[eventType]; //key为eventType的value
if (d != null && d.GetType() != callBack.GetType())
{
throw new Exception(string.Format("尝试为事件{0}添加不同的委托,当前事件所对应的委托为{1},要添加的委托类型为{2}", eventType, d.GetType(), callBack.GetType())); } //要添加的委托和已经存在委托一致了
//关联一下
m_EventTable[eventType] =(CallBack) m_EventTable[eventType] + callBack;
} //无参的移除监听
public static void RemoveListener(EventType eventType, CallBack callBack)
{
if (m_EventTable.ContainsKey(eventType))
{
Delegate d = m_EventTable[eventType];
if (d == null) //为空表示事件码没有对应的委托
{
throw new Exception(string.Format("移除监听错误,事件{0}没有对应委托", eventType));
}
else if (d.GetType() != callBack.GetType()) //判断委托类型是否一致,一致才能移除
{
throw new Exception(String.Format("移除监听错误,尝试为事件{0}移除不同类型的委托,当前委托类型为{1},要移除的委托类型为{2}", eventType, d.GetType(), callBack.GetType()));
}
}
else //不存在事件码
{
throw new Exception(String.Format("移除监听错误,没有事件码{0}", eventType));
} //可以移除
m_EventTable[eventType] = (CallBack)m_EventTable[eventType] - callBack; if (m_EventTable[eventType] == null) //如果为空了,移除事件码
{
m_EventTable.Remove(eventType);
}
} //无参的广播监听
public static void Broadcast(EventType eventType)
{
//把事件码对应的委托取出来,调用一下委托
Delegate d;
if(m_EventTable.TryGetValue(eventType,out d)) //尝试获取该键的值
{
CallBack callBack = d as CallBack; //当参数不匹配时会强转失败
if(callBack != null)
{
callBack();
}
else
{
throw new Exception(String.Format("广播事件错误,事件{0}对应委托有不同的类型,", eventType));
}
}
}

ShowText

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class ShowText : MonoBehaviour
{
//给方法添加监听,设置一个码
//点击时只需要广播一下事件码
void Start()
{
gameObject.SetActive(false);
EventCenter.AddListener(EventType.ShowText, Show); //添加监听
} private void OnDestroy()
{
EventCenter.RemoveListener(EventType.ShowText, Show);
} private void Show()
{
gameObject.SetActive(true);
GetComponent<Text>().text = "Hello";
}
}

BtnClick

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI; public class BtnClick : MonoBehaviour
{ void Start()
{
//Lambda表达式
GetComponent<Button>().onClick.AddListener(()=>{
EventCenter.Broadcast(EventType.ShowText); //广播事件
});
}

有参和无参基本都是一样的,只是泛型那里添加个类型就好了,不再一一写了

Unity---高度解耦和的更多相关文章

  1. 可复用且高度解耦的iOS用户统计实现

    http://www.cocoachina.com/ios/20160421/15912.html 本文为投稿文章,作者:编程小翁(简书) 用户统计 用户行为统计(User Behavior Stat ...

  2. 通过Unity依赖注入

    前言 Unity容器的思想起始于我在为Web Client Sofitware Factory项目工作的时候,微软的patterns&practices团队已经使用依赖注入的概念好几年了在那时 ...

  3. Unity学习系列一简介

    一.简介 Unity的目标是为了提升"依赖注入"的思想,去建立更加松耦合的系统.patterns & practices 小组在那个时候实现DI的方式和我们现在认为的DI有 ...

  4. 15个具有高度影响力的Apache开源项目

    自1999年创立以来,Apache软件基金会如今已成了众多重要的开源软件项目之家.其中成功的项目有Geronimo,有Tomcat,有Hadoop,有如今成了大数据王国关键车毂的分布式计算系统. 虽然 ...

  5. ASP.NET CORE 内置的IOC解读及使用

    在我接触IOC和DI 概念的时候是在2016年有幸倒腾Java的时候第一次接触,当时对这两个概念很是模糊:后来由于各种原因又回到.net 大本营,又再次接触了IOC和DI,也算终于搞清楚了IOC和DI ...

  6. zerojs! 造出最好的 CMS 轮子

    zerojs是一个基于nodejs.angularjs.git的CMS.在它之上可以继续开发出博客.论坛.wiki等类似的内容管理型系统. 拥抱开发者和社区 层次清晰,高度解耦.前后端即使分开也都是完 ...

  7. Flux 普及读本

    话说当时做 APP 时,三月不知肉味,再次将眼光投放前端,有种天上一天,地下一年的感觉. Flux 是一种思想 了解的最好方式当然是看Flux官方文档了.React 中文站点也能找到对应的翻译版本,但 ...

  8. 安卓v7支持包下的ListView替代品————RecyclerView

    RecyclerView这个控件也出来很久了,相信大家也学习的差不多了,如果还没学习的,或许我可以带领大家体验一把这个艺术般的控件. 项目已经同步至github:https://github.com/ ...

  9. 【转】理解 PHP 依赖注入 | Laravel IoC容器

    Laravel框架的依赖注入确实很强大,并且通过容器实现依赖注入可以有选择性的加载需要的服务,减少初始化框架的开销,下面是我在网上看到的一个帖子,写的很好拿来与大家分享,文章从开始按照传统的类设计数据 ...

  10. Vue 入门指南

    英文:http://vuejs.org/guide/index.html 介绍 vue.js 是用来构建web应用接口的一个库 技术上,Vue.js 重点集中在MVVM模式的ViewModel层,它连 ...

随机推荐

  1. Vulkan Tutorial 04 理解Validation layers

    操作系统:Windows8.1 显卡:Nivida GTX965M 开发工具:Visual Studio 2017 What are validation layers? Vulkan API的设计核 ...

  2. 专利系统数据库连接出现 base-64字符串中的无效字符 错误

    错误提示如图: 解决方法: 1.进注册表修改如下 2.进入系统配置页面http://10.10.0.70/eaf/init 对数据库进行重新配置 3.若不行再将如下密码修改一下 重启IIS生效

  3. 安装express.js(NODEJS框架)

    express.js是nodejs的一个MVC开发框架,并且支持jade等多种模板.下面简单来说说express的安装和app.js文件的配置,然后在今后的教程中一步一步使用express.js搭建个 ...

  4. tomcat在linux服务器上部署应用

    连接服务器 服务器地址:xxx.xxx.xxx.xxx 用户名:xxxx 密码:xxxx 进入到服务器中的tomcat路径,关闭服务器,例如 路径:/opt/wzgcyth/apache-tomcat ...

  5. Codeforces 1120C Compress String(DP)

    题意:给你一个字符串,有2种消除方式:1:消除一个单独的字母,代价为a.2:s[j]到s[k]是s[1]到s[j - 1]的子串,那么s[j]到s[k]可以消除,代价为b,问最小的代价. 思路:官方题 ...

  6. Quick Find

    --------------------siwuxie095                                 Quick Find         这里介绍并查集的一种实现思路:Qui ...

  7. Luogu 4473 [国家集训队]飞飞侠

    BZOJ 2143 新技能:并查集优化最短路. 暴力最短路是$O(n^4)$的,然后拿个线段树优化一下连边就$O($能过$)$了. 但是这样都太慢了. 我们考虑一个点如果之前被更新过了,那么之后就不会 ...

  8. Luogu 2868 [USACO07DEC]观光奶牛Sightseeing Cows

    01分数规划复习. 这东西有一个名字叫做最优比率环. 首先这个答案具有单调性,我们考虑如何检验. 设$\frac{\sum_{i = 1}^{n}F_i}{\sum_{i = 1}^{n}T_i} = ...

  9. c# enum遍历

    public enum Suit { Spades, Hearts, Clubs, Diamonds } //遍历valueforeach (Suit suit in (Suit[]) Enum.Ge ...

  10. layui之弹出层--从父窗口传递数据到子窗口

    原文链接:https://blog.csdn.net/Code_shadow/article/details/80524633 var Index = layer.open({ title: &quo ...