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

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

相关文章

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

Nginx location匹配模式与规则详解

《Nginxlocation匹配模式与规则详解》:本文主要介绍Nginxlocation匹配模式与规则,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、环境二、匹配模式1. 精准模式2. 前缀模式(不继续匹配正则)3. 前缀模式(继续匹配正则)4. 正则模式(大

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

SpringRetry重试机制之@Retryable注解与重试策略详解

《SpringRetry重试机制之@Retryable注解与重试策略详解》本文将详细介绍SpringRetry的重试机制,特别是@Retryable注解的使用及各种重试策略的配置,帮助开发者构建更加健... 目录引言一、SpringRetry基础知识二、启用SpringRetry三、@Retryable注解

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

SpringKafka错误处理(重试机制与死信队列)

《SpringKafka错误处理(重试机制与死信队列)》SpringKafka提供了全面的错误处理机制,通过灵活的重试策略和死信队列处理,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、Spring Kafka错误处理基础二、配置重试机制三、死信队列实现四、特定异常的处理策略五