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

  • 脚本型:脚本型的程序包括最早的批处理文件以及使用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实现来实现上述的业务逻辑,先看流程图:

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:passself.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:passself.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 = eventManagerdef 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") #订阅者1listner2 = 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】详解事件驱动event实现相关推荐

  1. python input与返回值-Python 详解基本语法_函数_返回值

    Python 详解基本语法 概要: 函数的返回值是函数重要的组成部分.函数的根本在于实现程序的部分功能,所以很多时候我们需要将函数执行后的结果返回给程序再由程序作出进一步的操作.可以说是函数的返回值令 ...

  2. python选择排序从大到小_经典排序算法和Python详解之(一)选择排序和二元选择排序...

    本文源自微信公众号[Python编程和深度学习]原文链接:经典排序算法和Python详解之(一)选择排序和二元选择排序,欢迎扫码关注鸭! 扫它!扫它!扫它 排序算法是<数据结构与算法>中最 ...

  3. qpython能使用json吗l_[python] 详解Python在使用JSON时需要注意的编码问题

    Python 中的字符编码 在 Python3 中, 字符 在内存中是使用 Unicode 存储的, 常规的字符使用 两个字节 表示, 一些很生僻的字符就需要 四个字节. 默认使用 Unicode 存 ...

  4. map函数的用法python,详解Python map函数及Python map()函数的用法

    python map函数 map()函数 map()是 Python 内置的高阶函数,它接收一个函数 f 和一个 list,并通过把函数 f 依次作用在 list 的每个元素上,得到一个新的 list ...

  5. 名片管理系统python详解_详解Python做一个名片管理系统

    详解Python做一个名片管理系统 来源:中文源码网    浏览: 次    日期:2019年11月5日 [下载文档:  详解Python做一个名片管理系统.txt ] (友情提示:右键点上行txt文 ...

  6. 使用Swig转C++为Python详解

    Swig是一个可以使得不同语言之间可以相互调用的工具,今天记录一下使用Swig将C++转为Python的方法. 首先从官网下载Swig,官网网址www.swig.org,下载后解压压缩包,并将swig ...

  7. 一看就懂!【英雄联盟锐雯】与 Python 详解设计模式之门面模式

    [网络配图] 设计模式(Design Pattern)是一套被反复使用.多数人知晓的.经过分类的.代码设计经验的总结.使用设计模式的目的:为了代码可重用性.让代码更容易被他人理解.保证代码可靠性.设计 ...

  8. 用 Python 详解《英雄联盟》游戏取胜的重要因素!

    作者 | DeepHub IMBA 责编 | 王晓曼 介绍 在过去的几年里,电子竞技社区发展迅速,曾经只是休闲娱乐的电子竞技产业,到2022年有望创造18亿美元的收入. 虽然在这个生态系统中有很多电子 ...

  9. JavaScript 运行机制详解:Event Loop

    转自: http://www.ruanyifeng.com/blog/2014/10/event-loop.html 一.为什么JavaScript是单线程? JavaScript语言的一大特点就是单 ...

最新文章

  1. Oracle 表分区删除操作
  2. Makefile_07:Make 的工作方式
  3. 读取文档时出现问题129_springboot读取配置文件的自定义内容时出现中文乱码
  4. jedisPool.getResource()方法长时间无响应并且不报错
  5. SpringCloud与Seata分布式事务初体验
  6. Linux环境编程导引
  7. PAT1119 Pre- and Post-order Traversals(先序后序求中序)
  8. 如果网站的 Cookie 特别多特别大,会发生什么情况?
  9. python论文题目_有关利用python获取网页, 以及KDD近几年论文标题与摘要链接
  10. 省赛来了(组合数)nyoj158
  11. linux下安装交叉编译器
  12. java二叉树删除节点
  13. linux minerd 进程,linux中了minerd之后的完全清理过程(详解)
  14. Python索引 说明
  15. 文件模式为 rw-r r linux,在Linux中,file.sh文件的权限为-rw-r--r-
  16. EFI Driver Model(上)
  17. 8000字前端性能优化技巧总结!(全面)
  18. 【obs-studio开源项目从入门到放弃】obs-studio项目简介和架构
  19. apiCloud app调用浏览器打开网页的方法
  20. Elang之ETS系列函数功能与用法详解

热门文章

  1. Invalid bound statement (not found): com.qf.zsm.springbootdynamicdatasource.mapper.TestMapper.insert
  2. tomcat启动时提示端口被占用解决办法
  3. IDEA端口被占用解决方案
  4. 分布式(集群)的基本概念以及分布式的应用场景
  5. jsEs6 map,symble
  6. ArraryList与HashSet——流下了不学无术的眼泪
  7. 【软件测试】软件测试方法之黑盒测试方法和白盒测试
  8. android 耳返解决方案,vivo又出招手机音频领域:联合全民K歌发布首个安卓实时耳返方案...
  9. 前端(一)html、css-学习笔记整理
  10. vim粘贴从其他地方复制过来的内容时多出来其他字符比如反斜杠或空格