作为python小白,学习量化交易的曲线是非常陡峭的,唯一好的办法就是一点点啃代码。以下代码案例来自vnpy的引擎代码。

  1. # encoding: UTF-8
  1. #定义时间事件
  1. EVENT_TIMER = eTimer
  2.  
  3. # 系统模块
  4. from Queue import Queue, Empty
  5. from threading import Thread
  6. from time import sleep
  7.  
  8. ########################################################################
  9. class EventEngine(object):
  10. """
  11. 事件驱动引擎
  12.  
  13. 事件驱动引擎中所有的变量都设置为了私有,这是为了防止不小心
  14. 从外部修改了这些变量的值或状态,导致bug。
  15.  
  16. 变量说明
  17. __queue:私有变量,事件队列
  18. __active:私有变量,事件引擎开关
  19. __thread:私有变量,事件处理线程
  20. __timer:私有变量,计时器
  21. __handlers:私有变量,事件处理函数字典
  22.  
  23. 方法说明
  24. __run: 私有方法,事件处理线程连续运行用
  25. __process: 私有方法,处理事件,调用注册在引擎中的监听函数
  26. __onTimer:私有方法,计时器固定事件间隔触发后,向事件队列中存入计时器事件
  27. start: 公共方法,启动引擎
  28. stop:公共方法,停止引擎
  29. register:公共方法,向引擎中注册监听函数
  30. unregister:公共方法,向引擎中注销监听函数
  31. put:公共方法,向事件队列中存入新的事件
  32.  
  33. 事件监听函数必须定义为输入参数仅为一个event对象,即:
  34.  
  35. 函数
  36. def func(event)
  37. ...
  38.  
  39. 对象方法
  40. def method(self, event)
  41. ...
  42.  
  43. """
  44.  
  45. #----------------------------------------------------------------------
  46. def __init__(self):
  47. """初始化事件引擎"""
  48. # 事件队列
  49. self.__queue = Queue()
  50.  
  51. # 事件引擎开关
  52. self.__active = False
  53.  
  54. # 事件处理线程
  55. self.__thread = Thread(target = self.__run)
  56.  
  57. # 计时器,用于触发计时器事件
  58. self.__timer = QTimer()
  59. self.__timer.timeout.connect(self.__onTimer)
  60.  
  61. # 这里的__handlers是一个字典,用来保存对应的事件调用关系
  62. # 其中每个键对应的值是一个列表,列表中保存了对该事件进行监听的函数功能
  63. self.__handlers = {}
  64.  
  65. #----------------------------------------------------------------------
  66. def __run(self):
  67. """引擎运行"""
  68. while self.__active == True:
  69. try:
  70. event = self.__queue.get(block = True, timeout = 1) # 获取事件的阻塞时间设为1秒
  71. self.__process(event)
  72. except Empty:
  73. pass
  74.  
  75. #----------------------------------------------------------------------
  76. def __process(self, event):
  77. """处理事件"""
  78. # 检查是否存在对该事件进行监听的处理函数
  79. if event.type_ in self.__handlers:
  80. # 若存在,则按顺序将事件传递给处理函数执行
  81. [handler(event) for handler in self.__handlers[event.type_]]
  82.  
  83. # 以上语句为Python列表解析方式的写法,对应的常规循环写法为:
  84. #for handler in self.__handlers[event.type_]:
  85. #handler(event)
  86.  
  87. #----------------------------------------------------------------------
  88. def __onTimer(self):
  89. """向事件队列中存入计时器事件"""
  90. # 创建计时器事件
  91. event = Event(type_=EVENT_TIMER)
  92.  
  93. # 向队列中存入计时器事件
  94. self.put(event)
  95.  
  96. #----------------------------------------------------------------------
  97. def start(self):
  98. """引擎启动"""
  99. # 将引擎设为启动
  100. self.__active = True
  101.  
  102. # 启动事件处理线程
  103. self.__thread.start()
  104.  
  105. # 启动计时器,计时器事件间隔默认设定为1秒
  106. self.__timer.start(1000)
  107.  
  108. #----------------------------------------------------------------------
  109. def stop(self):
  110. """停止引擎"""
  111. # 将引擎设为停止
  112. self.__active = False
  113.  
  114. # 停止计时器
  115. self.__timer.stop()
  116.  
  117. # 等待事件处理线程退出
  118. self.__thread.join()
  119.  
  120. #----------------------------------------------------------------------
  121. def register(self, type_, handler):
  122. """注册事件处理函数监听"""
  123. # 尝试获取该事件类型对应的处理函数列表,若无则创建
  124. try:
  125. handlerList = self.__handlers[type_]
  126. except KeyError:
  127. handlerList = []
  128. self.__handlers[type_] = handlerList
  129.  
  130. # 若要注册的处理器不在该事件的处理器列表中,则注册该事件
  131. if handler not in handlerList:
  132. handlerList.append(handler)
  133.  
  134. #----------------------------------------------------------------------
  135. def unregister(self, type_, handler):
  136. """注销事件处理函数监听"""
  137. # 尝试获取该事件类型对应的处理函数列表,若无则忽略该次注销请求
  138. try:
  139. handlerList = self.__handlers[type_]
  140.  
  141. # 如果该函数存在于列表中,则移除
  142. if handler in handlerList:
  143. handlerList.remove(handler)
  144.  
  145. # 如果函数列表为空,则从引擎中移除该事件类型
  146. if not handlerList:
  147. del self.__handlers[type_]
  148. except KeyError:
  149. pass
  150.  
  151. #----------------------------------------------------------------------
  152. def put(self, event):
  153. """向事件队列中存入事件"""
  154. self.__queue.put(event)
  155.  
  156. ########################################################################
  157. class EventEngine2(object):
  158. """
  159. 计时器使用python线程的事件驱动引擎
  160. """
  161.  
  162. #----------------------------------------------------------------------
  163. def __init__(self):
  164. """初始化事件引擎"""
  165. # 事件队列
  166. self.__queue = Queue()
  167.  
  168. # 事件引擎开关
  169. self.__active = False
  170.  
  171. # 事件处理线程
  172. self.__thread = Thread(target = self.__run)
  173.  
  174. # 计时器,用于触发计时器事件
  175. self.__timer = Thread(target = self.__runTimer)
  176. self.__timerActive = False # 计时器工作状态
  177. self.__timerSleep = 10 # 计时器触发间隔(默认1秒)
  178.  
  179. # 这里的__handlers是一个字典,用来保存对应的事件调用关系
  180. # 其中每个键对应的值是一个列表,列表中保存了对该事件进行监听的函数功能
  181. self.__handlers = {}
  182.  
  183. #----------------------------------------------------------------------
  184. def __run(self):
  185. """引擎运行"""
  186. print 'run'
  187. while self.__active == True:
  188. try:
  189. event = self.__queue.get(block = True, timeout = 1) # 获取事件的阻塞时间设为1秒
  190. #self.__process(event)
  191. except Empty:
  192. pass
  193.  
  194. #----------------------------------------------------------------------
  195. def __process(self, event):
  196. """处理事件"""
  197.  
  198. # 检查是否存在对该事件进行监听的处理函数
  199. if event.type_ in self.__handlers:
  200. # 若存在,则按顺序将事件传递给处理函数执行
  201. [handler(event) for handler in self.__handlers[event.type_]]
  202.  
  203. # 以上语句为Python列表解析方式的写法,对应的常规循环写法为:
  204. #for handler in self.__handlers[event.type_]:
  205. # handler(event)
  206.  
  207. #----------------------------------------------------------------------
  208. def __runTimer(self):
  209. """运行在计时器线程中的循环函数"""
  210.  
  211. while self.__timerActive:
  212. # 创建计时器事件
  213. event = Event(type_=EVENT_TIMER)
  214. # 向队列中存入计时器事件
  215. self.put(event)
  216. print 'runtimer%s',str(datetime.now())
  217. # 等待
  218. sleep(self.__timerSleep)
  219.  
  220. #----------------------------------------------------------------------
  221. def start(self):
  222. """引擎启动"""
  223. # 将引擎设为启动
  224. self.__active = True
  225.  
  226. # 启动事件处理线程
  227. self.__thread.start()
  228.  
  229. # 启动计时器,计时器事件间隔默认设定为1秒
  230. self.__timerActive = True
  231. self.__timer.start()
  232.  
  233. #----------------------------------------------------------------------
  234. def stop(self):
  235. """停止引擎"""
  236. # 将引擎设为停止
  237. self.__active = False
  238.  
  239. # 停止计时器
  240. self.__timerActive = False
  241. self.__timer.join()
  242.  
  243. # 等待事件处理线程退出
  244. self.__thread.join()
  245.  
  246. #----------------------------------------------------------------------
  247. def register(self, type_, handler):
  248. """注册事件处理函数监听"""
  249. # 尝试获取该事件类型对应的处理函数列表,若无则创建
  250. try:
  251. handlerList = self.__handlers[type_]
  252. except KeyError:
  253. handlerList = []
  254. self.__handlers[type_] = handlerList
  255.  
  256. # 若要注册的处理器不在该事件的处理器列表中,则注册该事件
  257. if handler not in handlerList:
  258. handlerList.append(handler)
  259.  
  260. #----------------------------------------------------------------------
  261. def unregister(self, type_, handler):
  262. """注销事件处理函数监听"""
  263. # 尝试获取该事件类型对应的处理函数列表,若无则忽略该次注销请求
  264. try:
  265. handlerList = self.__handlers[type_]
  266.  
  267. # 如果该函数存在于列表中,则移除
  268. if handler in handlerList:
  269. handlerList.remove(handler)
  270.  
  271. # 如果函数列表为空,则从引擎中移除该事件类型
  272. if not handlerList:
  273. del self.__handlers[type_]
  274. except KeyError:
  275. pass
  276.  
  277. #----------------------------------------------------------------------
  278. def put(self, event):
  279. """向事件队列中存入事件"""
  280. self.__queue.put(event)
  281.  
  282. ########################################################################
  283. class Event:
  284. """事件对象"""
  285.  
  286. #----------------------------------------------------------------------
  287. def __init__(self, type_=None):
  288. """Constructor"""
  289. self.type_ = type_ # 事件类型
  290. self.dict_ = {} # 字典用于保存具体的事件数据
  291.  
  292. #----------------------------------------------------------------------
  293. def test():
  294. """测试函数"""
  295. import sys
  296. from datetime import datetime
  297. from PyQt4.QtCore import QCoreApplication
  298.  
  299. def simpletest(event):
  300. print u'处理每秒触发的计时器事件:%s' % str(datetime.now())
  301. #app = QCoreApplication(sys.argv)
  302.  
  303. ee = EventEngine2()
  304. ee.register(EVENT_TIMER, simpletest)
  305. ee.start()
  306. #app.exec_()
  307.  
  308. # 直接运行脚本可以进行测试
  309. if __name__ == '__main__':
  310. test()

  

Python建立时间事件引擎原理剖析的更多相关文章

  1. MapReduce/Hbase进阶提升(原理剖析、实战演练)

    什么是MapReduce? MapReduce是一种编程模型,用于大规模数据集(大于1TB)的并行运算.概念"Map(映射)"和"Reduce(归约)",和他们 ...

  2. 剖析Qt的事件机制原理

    版权声明 请尊重原创作品.转载请保持文章完整性,并以超链接形式注明原始作者“tingsking18”和主站点地址,方便其他朋友提问和指正. QT源码解析(一) QT创建窗口程序.消息循环和WinMai ...

  3. Python字符串原理剖析------万恶的+号

    字符串原理剖析pyc文件,执行python代码时,如果导入了其他的.py文件,那么执行过程中会自动生成一个与其同名的.pyc文件,该文件就是python解释器变异之后产生的字节码 PS:代码经过编译可 ...

  4. 推荐《深入浅出深度学习原理剖析与python实践》PDF+代码

    <深入浅出深度学习原理剖析与Python实践>介绍了深度学习相关的原理与应用,全书共分为三大部分,第一部分主要回顾了深度学习的发展历史,以及Theano的使用:第二部分详细讲解了与深度学习 ...

  5. 深入浅出深度学习:原理剖析与python实践_黄安埠(著) pdf

    深入浅出深度学习:原理剖析与python实践 目录: 第1 部分 概要 1 1 绪论 2 1.1 人工智能.机器学习与深度学习的关系 3 1.1.1 人工智能——机器推理 4 1.1.2 机器学习—— ...

  6. python生成器原理剖析

    python生成器原理剖析 函数的调用满足"后进先出"的原则,也就是说,最后被调用的函数应该第一个返回,函数的递归调用就是一个经典的例子.显然,内存中以"后进先出&quo ...

  7. 开源 serverless 产品原理剖析 - Kubeless

    背景 Serverless 架构的出现让开发者不用过多地考虑传统的服务器采购.硬件运维.网络拓扑.资源扩容等问题,可以将更多的精力放在业务的拓展和创新上. 随着 serverless 概念的深入人心, ...

  8. 基本功 | Litho的使用及原理剖析

    1. 什么是Litho? Litho是Facebook推出的一套高效构建Android UI的声明式框架,主要目的是提升RecyclerView复杂列表的滑动性能和降低内存占用.下面是Litho官网的 ...

  9. python 3 mysql 索引原理与慢查询优化

    python 3 mysql 索引原理与慢查询优化 一 介绍 为何要有索引? 一般的应用系统,读写比例在10:1左右,而且插入操作和一般的更新操作很少出现性能问题,在生产环境中,我们遇到最多的,也是最 ...

随机推荐

  1. Rabbitmq 安装后采坑

    一.接手项目 接手项目后,按别人说的先安装什么,后安装什么然后就可以用了,也不去看什么.先开始安装的是otp_win64_19.1工具包和rabbitmq-server-3.6.5服务端,在win10 ...

  2. Python第8天

    zip() 拉链方法 max(字典) 默认比较字典的key,不同类型的数据不能比较,只要可以被for迭代即可 利用zip与max(字典)共同使用 ord() — chr()    ascii码表数字与 ...

  3. 数据库分库分表(sharding)系列

    数据库分库分表(sharding)系列     目录; (一) 拆分实施策略和示例演示 (二) 全局主键生成策略 (三) 关于使用框架还是自主开发以及sharding实现层面的考量 (四) 多数据源的 ...

  4. 在虚拟机上运行zookeeper的过程中,xshell连接不上虚拟机

    之后网上的各种方法,都没用 如图,就是第一个网卡配置文件.下面又具体的详解.网上有说让把Onboot改成yes,而我的本来就是yes,就没该, 之后我对比了我原来电脑你虚拟机的网卡配置文件,我发现on ...

  5. node环境

    下载教程:http://www.runoob.com/nodejs/nodejs-install-setup.html 选择版本下载:https://nodejs.org/en/download/ 输 ...

  6. javaScript:压缩图片并上传

    html代码: <input id="file" type="file" name="filesName"> js代码: var ...

  7. Django08-批量创建数据

    通过views.py文件中创建 第1种方法循环创建数据, 这种方法不推荐,因为每一次循环都会连接一次数据库,效率较慢 def user_list(request): user_all = models ...

  8. mysql 定时计划任务 wish 按照id分组定时循环启动

    SELECT count(*) FROM wish_sellers_in;UPDATE  wish_sellers_in  SET act_status =0 WHERE  id >=1 AND ...

  9. 记一次laravel-jwt修改黑名单所用redis数据库

    场景是这样的,我用tymon/jwt包做鉴权.jwt是自编码token,过期前想要强制失效只能将其加入黑名单中,黑名单一般用缓存存储. 但会有一个问题,若某种意外情况不小心执行了php aritsan ...

  10. linux下打压缩解压

    tar -c: 建立压缩档案-x:解压-t:查看内容-r:向压缩归档文件末尾追加文件-u:更新原压缩包中的文件 这五个是独立的命令,压缩解压都要用到其中一个,可以和别的命令连用但只能用其中一个.下面的 ...