Python自学成才之路 分布式计算解决方案actor

2023-10-09 21:59

本文主要是介绍Python自学成才之路 分布式计算解决方案actor,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

以下内容来自于cookbook,个人觉得这篇文章对于设计分布式计算任务有一定的借鉴意义,感兴趣的同学可以阅读原文:
https://python3-cookbook.readthedocs.io/zh_CN/latest/c12/p10_defining_an_actor_task.html

actor模式是一种最古老的也是最简单的并行和分布式计算解决方案。一个actor就是一个并发执行的任务,只是简单的执行发送给它的消息任务。actor之间的通信是单向和异步的。因此,消息发送者不知道消息是什么时候被发送, 也不会接收到一个消息已被处理的回应或通知。

使用线程加队列可以定义一个actor:

from queue import Queue
from threading import Thread, Event
import timeclass ActorExit(Exception):passclass Actor:def __init__(self):self._mailbox = Queue()def send(self, msg):'''发送消息 '''self._mailbox.put(msg)def recv(self):'''接受消息'''msg = self._mailbox.get()if msg is ActorExit:raise ActorExit()return msgdef close(self):'''关闭actor'''self.send(ActorExit)def start(self):'''启动actor'''self._terminated = Event()t = Thread(target=self._bootstrap)t.daemon = Truet.start()def _bootstrap(self):try:self.run()except ActorExit:passfinally:self._terminated.set()def join(self):self._terminated.wait()def run(self):'''消费者线程的run方法'''while True:msg = self.recv()class PrintActor(Actor):def run(self):while True:msg = self.recv()print('Got:', msg)p = PrintActor()
p.start()
time.sleep(1)
p.send('Hello')
time.sleep(2)
p.send('World')
time.sleep(1)
p.close()
p.join()输出:
Got: Hello
Got: World

这个案例是这样的,先定义了一个actor,然后通过继承actor定义了PrintActor,并重写了其中的run方法。

这个actor中的每个方法起到什么作用,并如何实现的?
Start:创建self._terminated信号(关于event我前面的文章有讲解),定义一个线程,将现场设置为守护线程,并启动线程。
_bootstrap:线程启动后会执行这个方法,并启动里面的run方法。
Run:这里的run方法被子类重写了,子类通过recv不断的获取队列中的消息。
Recv:获取队列中的消息,并返回消息,如果消息类型是ActorExit,则抛出ActorExit异常。
Send:往队列中写入消息
Close:发送一个ActorExit类型的消息,recv接受到这个消息后,抛出ActorExit异常,run方法结束,bootstrap中捕获这个异常后,将self._terminated信号修改为true。
Join:如果self._terminated为false线程阻塞。

思考:在程序最后面加了一个p.join(),为何要加这个呢?
因为close方法只是发送了一个ActorExit类型的消息到队列,而消费队列的线程是一个守护线程,如果close后面没有任何需要执行的代码,则主线程就结束了,子线程也会随着结束,那么很有可能‘world’这条消息也没消费到,子线程就退出了,所以加个p.join()是为了主线程被阻塞,知道子线程消费到ActorExit类型的消息,将self._terminated设置为true才会唤醒主线程,主线程退出,子线程也退出,此时队列中的消息肯定是已经消费完了。


上面这个案例中,actor只是简单的模拟了一个队列传递消息的例子,实际上这些消息还可以是一个函数,比如下面这个案例:


from threading import Event
class Result:def __init__(self):self._evt = Event()self._result = Nonedef set_result(self, value):self._result = valueself._evt.set()def result(self):self._evt.wait()return self._resultclass Worker(Actor):def submit(self, func, *args, **kwargs):r = Result()self.send((func, args, kwargs, r))return rdef run(self):while True:func, args, kwargs, r = self.recv()r.set_result(func(*args, **kwargs))def add(a, b):return a+bworker = Worker()
worker.start()
r = worker.submit(add, 2, 3)
worker.close()
worker.join()
print(r.result())
输出:
5

这个案例中,worker继承了actor,在submit方法中,将函数和函数的参数作为一个元祖发送到队列中,并返回一个Result实例,消费者线程拿到函数和参数后,执行这个函数,并将函数的返回值放到Result实例的_result属性上。_result属性值可以通过result()方法获取,这个方法是阻塞的,只有当线程将函数执行完并将函数结果通过set_result方法赋值给_result属性上时,result()方法才能得到返回结果,这里面的异步获取返回结果的过程是通过event()来控制的。

这种通过函数作为消息的方式来传递并由消费线程来执行的思想在一些分布式系统里面会经常用到。后面碰到这种案例再给兄弟们分享出来。



本人是做大数据开发的,在微信上开了个个人号,会经常在上面分享一些学习心得,原创文章都会首发到公众号上,感兴趣的盆友可以关注下哦!
在这里插入图片描述
备注:微信公众号搜索‘大数据入坑指南’

这篇关于Python自学成才之路 分布式计算解决方案actor的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python数据验证神器Pydantic库的使用和实践中的避坑指南

《Python数据验证神器Pydantic库的使用和实践中的避坑指南》Pydantic是一个用于数据验证和设置的库,可以显著简化API接口开发,文章通过一个实际案例,展示了Pydantic如何在生产环... 目录1️⃣ 崩溃时刻:当你的API接口又双叒崩了!2️⃣ 神兵天降:3行代码解决验证难题3️⃣ 深度

Python+FFmpeg实现视频自动化处理的完整指南

《Python+FFmpeg实现视频自动化处理的完整指南》本文总结了一套在Python中使用subprocess.run调用FFmpeg进行视频自动化处理的解决方案,涵盖了跨平台硬件加速、中间素材处理... 目录一、 跨平台硬件加速:统一接口设计1. 核心映射逻辑2. python 实现代码二、 中间素材处

python中的flask_sqlalchemy的使用及示例详解

《python中的flask_sqlalchemy的使用及示例详解》文章主要介绍了在使用SQLAlchemy创建模型实例时,通过元类动态创建实例的方式,并说明了如何在实例化时执行__init__方法,... 目录@orm.reconstructorSQLAlchemy的回滚关联其他模型数据库基本操作将数据添

Python实现快速扫描目标主机的开放端口和服务

《Python实现快速扫描目标主机的开放端口和服务》这篇文章主要为大家详细介绍了如何使用Python编写一个功能强大的端口扫描器脚本,实现快速扫描目标主机的开放端口和服务,感兴趣的小伙伴可以了解下... 目录功能介绍场景应用1. 网络安全审计2. 系统管理维护3. 网络故障排查4. 合规性检查报错处理1.

Python轻松实现Word到Markdown的转换

《Python轻松实现Word到Markdown的转换》在文档管理、内容发布等场景中,将Word转换为Markdown格式是常见需求,本文将介绍如何使用FreeSpire.DocforPython实现... 目录一、工具简介二、核心转换实现1. 基础单文件转换2. 批量转换Word文件三、工具特性分析优点局

Python中4大日志记录库比较的终极PK

《Python中4大日志记录库比较的终极PK》日志记录框架是一种工具,可帮助您标准化应用程序中的日志记录过程,:本文主要介绍Python中4大日志记录库比较的相关资料,文中通过代码介绍的非常详细,... 目录一、logging库1、优点2、缺点二、LogAid库三、Loguru库四、Structlogphp

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

idea粘贴空格时显示NBSP的问题及解决方案

《idea粘贴空格时显示NBSP的问题及解决方案》在IDEA中粘贴代码时出现大量空格占位符NBSP,可以通过取消勾选AdvancedSettings中的相应选项来解决... 目录1、背景介绍2、解决办法3、处理完成总结1、背景介绍python在idehttp://www.chinasem.cna粘贴代码,出

Python海象运算符:=的具体实现

《Python海象运算符:=的具体实现》海象运算符又称​​赋值表达式,Python3.8后可用,其核心设计是在表达式内部完成变量赋值并返回该值,从而简化代码逻辑,下面就来详细的介绍一下如何使用,感兴趣... 目录简介​​条件判断优化循环控制简化​推导式高效计算​正则匹配与数据提取​性能对比简介海象运算符

python项目环境切换的几种实现方式

《python项目环境切换的几种实现方式》本文主要介绍了python项目环境切换的几种实现方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 如何在不同python项目中,安装不同的依赖2. 如何切换到不同项目的工作空间3.创建项目