代码事件派发机制(观察者模式)

2023-10-07 20:52

本文主要是介绍代码事件派发机制(观察者模式),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

事件派发机制主要用来解决:

   代码解耦和维护,一般在代码中会要管理一些闭包函数  然后在指定的业务中触发运行闭包函数逻辑用了事件派发机制  就可以先把要处理的事件 挂在在一个事件管理类中  上面挂满要处理的闭包函数然后通过dispatch 出发要执行的任务 也就是闭包

1. PHP中实现

在 PHP 中,你可以使用事件派发机制来实现基于观察者模式的事件驱动开发。下面是一个简单的示例来展示如何实现事件派发机制:

<?phpclass EventDispatcher
{private $listeners = [];public function addListener($eventName, $listener){$this->listeners[$eventName][] = $listener;}public function dispatch($eventName, $data = null){if (isset($this->listeners[$eventName])) {foreach ($this->listeners[$eventName] as $listener) {call_user_func($listener, $data);}}}
}// 示例使用
$dispatcher = new EventDispatcher();// 添加事件监听器
$dispatcher->addListener('user.registered', function ($data) {echo "用户注册成功:{$data}\n";
});$dispatcher->addListener('user.loggedIn', function ($data) {echo "用户登录成功:{$data}\n";
});// 模拟触发事件
$dispatcher->dispatch('user.registered', 'user123');
$dispatcher->dispatch('user.loggedIn', 'user456');

在上面的示例中,我们创建了一个名为 EventDispatcher 的事件调度器类。该类包括 addListener() 方法用于添加事件监听器,和 dispatch() 方法用于触发事件并通知相应的监听器。

通过调用 addListener() 方法,我们可以为特定的事件名称注册一个或多个监听器。这些监听器是以匿名函数的方式定义的,用于处理事件被触发时的逻辑。

在 dispatch() 方法中,我们检查指定事件名称的监听器列表是否存在,然后依次调用每个监听器函数,并将事件数据作为参数传递给它们。

最后,在示例的最后部分,我们通过调用 dispatch() 方法模拟触发了两个事件,然后相应的监听器函数被调用并处理事件。

运行以上代码,将会输出如下信息:

用户注册成功:user123
用户登录成功:user456
这样就实现了一个简单的事件派发机制。在实际的应用中,你可以根据需要扩展和定制事件调度器,添加更多的事件和监听器,以实现更复杂的业务逻辑。

2 go语言中实现

在 Go 语言中,你可以使用通道(channel)和协程(goroutine)的特性来实现一个简单的事件派发机制。下面是一个示例代码:

package mainimport ("fmt""time"
)type Event struct {Name stringData interface{}
}type EventDispatcher struct {listeners map[string][]chan interface{}
}func NewEventDispatcher() *EventDispatcher {return &EventDispatcher{listeners: make(map[string][]chan interface{}),}
}func (ed *EventDispatcher) AddListener(eventName string, listener chan interface{}) {if _, ok := ed.listeners[eventName]; !ok {ed.listeners[eventName] = make([]chan interface{}, 0)}ed.listeners[eventName] = append(ed.listeners[eventName], listener)
}func (ed *EventDispatcher) Dispatch(eventName string, data interface{}) {if listeners, ok := ed.listeners[eventName]; ok {event := Event{Name: eventName,Data: data,}for _, listener := range listeners {go func(ch chan interface{}) {ch <- event}(listener)}}
}func main() {dispatcher := NewEventDispatcher()// 添加事件监听器listener1 := make(chan interface{})dispatcher.AddListener("event1", listener1)listener2 := make(chan interface{})dispatcher.AddListener("event2", listener2)// 模拟触发事件dispatcher.Dispatch("event1", "data1")dispatcher.Dispatch("event2", "data2")// 读取监听器收到的事件go func(ch chan interface{}) {for {event := <-chfmt.Println(event)}}(listener1)go func(ch chan interface{}) {for {event := <-chfmt.Println(event)}}(listener2)// 等待事件处理time.Sleep(time.Second)
}

在上述示例中,我们定义了 Event 结构体来表示事件,其中包含事件名称和事件数据。EventDispatcher 结构体用于管理事件监听器和事件派发。通过 AddListener 方法可以为指定事件名称注册一个事件监听器,而 Dispatch 方法用于派发事件并通知相应的监听器。

在 main 函数中,我们实例化了一个 EventDispatcher 对象,并添加了两个事件监听器,分别监听名为 “event1” 和 “event2” 的事件。

然后,我们使用 Dispatch 方法模拟触发了 “event1” 和 “event2” 事件,并将相关数据传递给监听器。

最后,我们在两个协程中创建了监听器函数,用于处理收到的事件。通过使用通道读取操作 <-ch,监听器函数可以不断等待事件的到来。

运行以上代码,将会得到类似如下的输出:

{event1 data1}
{event2 data2}
这样就实现了一个简单的事件派发机制。你可以根据需要扩展该事件派发器,增加更多事件和监听器,以满足具体的业务需求。同时需要注意,对于涉及并发的操作,需要进行适当的同步和错误处理。

这篇关于代码事件派发机制(观察者模式)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/160371

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

禁止平板,iPad长按弹出默认菜单事件

通过监控按下抬起时间差来禁止弹出事件,把以下代码写在要禁止的页面的页面加载事件里面即可     var date;document.addEventListener('touchstart', event => {date = new Date().getTime();});document.addEventListener('touchend', event => {if (new

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

Java ArrayList扩容机制 (源码解读)

结论:初始长度为10,若所需长度小于1.5倍原长度,则按照1.5倍扩容。若不够用则按照所需长度扩容。 一. 明确类内部重要变量含义         1:数组默认长度         2:这是一个共享的空数组实例,用于明确创建长度为0时的ArrayList ,比如通过 new ArrayList<>(0),ArrayList 内部的数组 elementData 会指向这个 EMPTY_EL

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能

代码随想录冲冲冲 Day39 动态规划Part7

198. 打家劫舍 dp数组的意义是在第i位的时候偷的最大钱数是多少 如果nums的size为0 总价值当然就是0 如果nums的size为1 总价值是nums[0] 遍历顺序就是从小到大遍历 之后是递推公式 对于dp[i]的最大价值来说有两种可能 1.偷第i个 那么最大价值就是dp[i-2]+nums[i] 2.不偷第i个 那么价值就是dp[i-1] 之后取这两个的最大值就是d

pip-tools:打造可重复、可控的 Python 开发环境,解决依赖关系,让代码更稳定

在 Python 开发中,管理依赖关系是一项繁琐且容易出错的任务。手动更新依赖版本、处理冲突、确保一致性等等,都可能让开发者感到头疼。而 pip-tools 为开发者提供了一套稳定可靠的解决方案。 什么是 pip-tools? pip-tools 是一组命令行工具,旨在简化 Python 依赖关系的管理,确保项目环境的稳定性和可重复性。它主要包含两个核心工具:pip-compile 和 pip

【编程底层思考】垃圾收集机制,GC算法,垃圾收集器类型概述

Java的垃圾收集(Garbage Collection,GC)机制是Java语言的一大特色,它负责自动管理内存的回收,释放不再使用的对象所占用的内存。以下是对Java垃圾收集机制的详细介绍: 一、垃圾收集机制概述: 对象存活判断:垃圾收集器定期检查堆内存中的对象,判断哪些对象是“垃圾”,即不再被任何引用链直接或间接引用的对象。内存回收:将判断为垃圾的对象占用的内存进行回收,以便重新使用。