python事件驱动event实现详解


Posted in Python onNovember 21, 2018

所有的计算机程序都可以大致分为两类:脚本型(单次运行)和连续运行型(直到用户主动退出)。

脚本型:脚本型的程序包括最早的批处理文件以及使用Python做交易策略回测等等,这类程序的特点是在用户启动后会按照编程时设计好的步骤一步步运行,所有步骤运行完后自动退出。

连续运行型:连续运行型的程序包含了操作系统和绝大部分我们日常使用的软件等等,这类程序启动后会处于一个无限循环中连续运行,直到用户主动退出时才会结束。

一、连续运行型程序

我们要开发的交易系统就是属于连续运行型程序,而这种程序根据其计算逻辑的运行机制不同,又可以粗略的分为时间驱动和事件驱动两种。

1.1 时间驱动

时间驱动的程序逻辑相对容易设计,简单来说就是让电脑每隔一段时间自动做一些事情。这个事情本身可以很复杂、包括很多步骤,但这些步骤都是线性的,按照顺序一步步执行下来。

from time import sleep
def demo():
 print('BB')
while True:
 demo()
 sleep(1.0)

时间驱动的程序本质上就是每隔一段时间固定运行一次脚本。尽管脚本自身可以很长、包含非常多的步骤,但是我们可以看出这种程序的运行机制相对比较简单、容易理解。

时间驱动的程序在量化交易方面还存在一些其他的缺点:如浪费CPU的计算资源、实现异步逻辑复杂度高等等。

1.2 事件驱动

与时间驱动对应的就是事件驱动的程序:当某个新的事件被推送到程序中时,程序立即调用和这个事件相对应的处理函数进行相关的操作。

举个例子:

有些人喜欢的某个公众号,然后去关注这个公众号,哪天这个公众号发布了篇新的文章,没多久订阅者就会在微信里收到这个公众号推送的新消息,如果感兴趣就打开来阅读。

上面公众号例子可以翻译为,监听器(订阅者)监听了(关注了)事件源(公众号),当事件源的发送事件时(公众号发布文章),所有监听该事件的监听器(订阅者)都会接收到消息并作出响应(阅读文章)。

  • 公众号为事件源
  • 订阅者为事件监听器
  • 订阅者关注公众号,相当于监听器监听了事件源
  • 公众号发布文章这个动作为发送事件
  • 订阅者收到事件后,做出阅读文章的响应动作

事件驱动主要包含以下元素和操作函数:

1.2.1 元素

  • 事件源
  • 事件监听器
  • 事件对象

1.2.2 操作函数

  • 监听动作
  • 发送事件
  • 调用监听器响应函数

现在用python实现来实现上述的业务逻辑,先看流程图:

python事件驱动event实现详解

1.2.3 EventManager事件管理类代码如下:

# -*- coding: utf-8 -*-
"""
Created on Tue Nov 13 13:51:31 2018

@author: 18665
"""
# 系统模块
from queue import Queue, Empty
from threading import *
########################################################################
class EventManager:
 #----------------------------------------------------------------------
 def __init__(self):
  """初始化事件管理器"""
  # 事件对象列表
  self.__eventQueue = Queue()
  # 事件管理器开关
  self.__active = False
  # 事件处理线程
  self.__thread = Thread(target = self.__Run)
  self.count = 0
  # 这里的__handlers是一个字典,用来保存对应的事件的响应函数
  # 其中每个键对应的值是一个列表,列表中保存了对该事件监听的响应函数,一对多
  self.__handlers = {}
 #----------------------------------------------------------------------
 def __Run(self):
  """引擎运行"""
  print('{}_run'.format(self.count))
  while self.__active == True:
   try:
    # 获取事件的阻塞时间设为1秒
    event = self.__eventQueue.get(block = True, timeout = 1) 
    self.__EventProcess(event)
   except Empty:
    pass
   self.count += 1
 #----------------------------------------------------------------------
 def __EventProcess(self, event):
  """处理事件"""
  print('{}_EventProcess'.format(self.count))
  # 检查是否存在对该事件进行监听的处理函数
  if event.type_ in self.__handlers:
   # 若存在,则按顺序将事件传递给处理函数执行
   for handler in self.__handlers[event.type_]:
    handler(event)
  self.count += 1
 #----------------------------------------------------------------------
 def Start(self):
  """启动"""
  print('{}_Start'.format(self.count))
  # 将事件管理器设为启动
  self.__active = True
  # 启动事件处理线程
  self.__thread.start()
  self.count += 1
 #----------------------------------------------------------------------
 def Stop(self):
  """停止"""
  print('{}_Stop'.format(self.count))
  # 将事件管理器设为停止
  self.__active = False
  # 等待事件处理线程退出
  self.__thread.join()
  self.count += 1
 #----------------------------------------------------------------------
 def AddEventListener(self, type_, handler):
  """绑定事件和监听器处理函数"""
  print('{}_AddEventListener'.format(self.count))
  # 尝试获取该事件类型对应的处理函数列表,若无则创建
  try:
   handlerList = self.__handlers[type_]
  except KeyError:
   handlerList = []
 self.__handlers[type_] = handlerList
  # 若要注册的处理器不在该事件的处理器列表中,则注册该事件
  if handler not in handlerList:
   handlerList.append(handler)
  print(self.__handlers)
  self.count += 1
 #----------------------------------------------------------------------
 def RemoveEventListener(self, type_, handler):
  """移除监听器的处理函数"""
  print('{}_RemoveEventListener'.format(self.count))
  try:
   handlerList = self.handlers[type_]
   # 如果该函数存在于列表中,则移除
   if handler in handlerList:
    handlerList.remove(handler)
   # 如果函数列表为空,则从引擎中移除该事件类型
   if not handlerList:
    del self.handlers[type_]
  except KeyError:
   pass
  self.count += 1
 #----------------------------------------------------------------------
 def SendEvent(self, event):
  """发送事件,向事件队列中存入事件"""
  print('{}_SendEvent'.format(self.count))
  self.__eventQueue.put(event)
  self.count += 1
########################################################################
"""事件对象"""
class Event:
 def __init__(self, type_=None):
  self.type_ = type_  # 事件类型
  self.dict = {}   # 字典用于保存具体的事件数据

1.2.4 测试代码

# -*- coding: utf-8 -*-
"""
Created on Tue Nov 13 13:50:45 2018

@author: 18665
"""

# encoding: UTF-8
import sys
from datetime import datetime
from threading import *
#sys.path.append('D:\\works\\TestFile')
#print(sys.path)
from eventManager import *

#事件名称 新文章
EVENT_ARTICAL = "Event_Artical"

#事件源 公众号
class PublicAccounts:
 def __init__(self,eventManager):
  self.__eventManager = eventManager

 def WriteNewArtical(self):
  #事件对象,写了新文章
  event = Event(type_=EVENT_ARTICAL)
  event.dict["artical"] = u'如何写出更优雅的代码\n'
  
  #发送事件
  self.__eventManager.SendEvent(event)
  print(u'公众号发送新文章\n')

#监听器 订阅者
class Listener:
 def __init__(self,username):
  self.__username = username

 #监听器的处理函数 读文章
 def ReadArtical(self,event):
  print(u'%s 收到新文章' % self.__username)
  print(u'正在阅读新文章内容:%s' % event.dict["artical"])

"""测试函数"""
#--------------------------------------------------------------------
def test():
 # 实例化监听器
 listner1 = Listener("thinkroom") #订阅者1
 listner2 = Listener("steve")  #订阅者2
 # 实例化事件操作函数
 eventManager = EventManager()

 #绑定事件和监听器响应函数(新文章)
 eventManager.AddEventListener(EVENT_ARTICAL, listner1.ReadArtical)
 eventManager.AddEventListener(EVENT_ARTICAL, listner2.ReadArtical)
 # 启动事件管理器,# 启动事件处理线程
 eventManager.Start()

 publicAcc = PublicAccounts(eventManager)
 timer = Timer(2, publicAcc.WriteNewArtical)
 timer.start()

if __name__ == '__main__':
 test()

通过eventManager可以实现事件触发的逻辑,当事件触发时,推送事件到线程里运行。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
详解Django框架中用context来解析模板的方法
Jul 20 Python
pygame游戏之旅 添加碰撞效果的方法
Nov 20 Python
Python数据类型之Number数字操作实例详解
May 08 Python
pygame实现五子棋游戏
Oct 29 Python
python循环输出三角形图案的例子
Nov 22 Python
基于python3实现倒叙字符串
Feb 18 Python
Python实现多线程下载脚本的示例代码
Apr 03 Python
Python中实现输入一个整数的案例
May 03 Python
python求numpy中array按列非零元素的平均值案例
Jun 08 Python
Python叠加矩形框图层2种方法及效果
Jun 18 Python
Python MNIST手写体识别详解与试练
Nov 07 Python
python 中的jieba分词库
Nov 23 Python
python程序封装为win32服务的方法
Mar 07 #Python
pygame游戏之旅 添加icon和bgm音效的方法
Nov 21 #Python
pygame游戏之旅 添加游戏暂停功能
Nov 21 #Python
使用50行Python代码从零开始实现一个AI平衡小游戏
Nov 21 #Python
pygame游戏之旅 调用按钮实现游戏开始功能
Nov 21 #Python
pygame游戏之旅 按钮上添加文字的方法
Nov 21 #Python
Face++ API实现手势识别系统设计
Nov 21 #Python
You might like
CodeIgniter php mvc框架 中国网站
2008/05/26 PHP
ThinkPHP Mobile使用方法简明教程
2014/06/18 PHP
Docker 如何布置PHP开发环境
2016/06/21 PHP
php array_slice 取出数组中的一段序列实例
2016/11/04 PHP
替换php字符串中的单引号为双引号的方法
2017/02/16 PHP
php使用curl_init()和curl_multi_init()多线程的速度比较详解
2018/08/15 PHP
php7 图形用户界面GUI 开发示例
2020/02/22 PHP
thinkphp诸多限制条件下如何getshell详解
2020/12/09 PHP
jquery获取table中的某行全部td的内容方法
2013/03/08 Javascript
javascript实现window.print()去除页眉页脚
2014/12/30 Javascript
js实现跨域的方法实例详解
2015/06/24 Javascript
jqGrid表格应用之新增与删除数据附源码下载
2015/12/02 Javascript
快速掌握Node.js环境的安装与运行方法
2016/02/16 Javascript
javascript实现无法关闭的弹框
2016/11/27 Javascript
JavaScript 复制对象与Object.assign方法无法实现深复制
2018/11/02 Javascript
详解JavaScript修改注册表的方法
2020/01/05 Javascript
用VsCode编辑TypeScript的实现方法
2020/05/07 Javascript
js实现查询商品案例
2020/07/22 Javascript
js实现滑动进度条效果
2020/08/21 Javascript
Vue Object.defineProperty及ProxyVue实现双向数据绑定
2020/09/02 Javascript
如何基于viewport vm适配移动端页面
2020/11/13 Javascript
[03:49]DOTA2 2015国际邀请赛中国区预选赛第二日现场百态
2015/05/27 DOTA
[02:00]DOTA2英雄COSPLAY闹市街头巡游助威2015国际邀请赛
2015/08/02 DOTA
python赋值操作方法分享
2013/03/23 Python
Python在Windows和在Linux下调用动态链接库的教程
2015/08/18 Python
Python简单实现子网掩码转换的方法
2016/04/13 Python
详解10个可以快速用Python进行数据分析的小技巧
2019/06/24 Python
在SQLite-Python中实现返回、查询中文字段的方法
2019/07/17 Python
Python3 使用selenium插件爬取苏宁商家联系电话
2019/12/23 Python
python3中datetime库,time库以及pandas中的时间函数区别与详解
2020/04/16 Python
完美解决pycharm 不显示代码提示问题
2020/06/02 Python
通过实例了解python__slots__使用方法
2020/09/14 Python
详解CSS透明opacity和IE各版本透明度滤镜filter的最准确用法
2016/12/20 HTML / CSS
医学专业本科毕业生自我鉴定
2013/12/28 职场文书
2014年资料员工作总结
2014/11/18 职场文书
SpringBoot工程下使用OpenFeign的坑及解决
2021/07/02 Java/Android