Python中使用asyncio 封装文件读写


Posted in Python onSeptember 11, 2016

前言

和网络 IO 一样,文件读写同样是一个费事的操作。

默认情况下,Python 使用的是系统的阻塞读写。这意味着在 asyncio 中如果调用了

f = file('xx')
f.read()

会阻塞事件循环。

本篇简述如何用 asyncio.Future 对象来封装文件的异步读写。

代码在 GitHub。目前仅支持 Linux。

阻塞和非阻塞

首先需要将文件的读写改为非阻塞的形式。在非阻塞情况下,每次调用 read 都会立即返回,如果返回值为空,则意味着文件操作还未完成,反之则是读取的文件内容。

阻塞和非阻塞的切换与操作系统有关,所以本篇暂时只写了 Linux 版本。如果有过 Unix 系统编程经验,会发现 Python 的操作是类似的。

flag = fcntl.fcntl(self.fd, fcntl.F_GETFL) 
if fcntl.fcntl(self.fd, fcntl.F_SETFL, flag | os.O_NONBLOCK) != 0: 
  raise OSError()

Future 对象

Future 对象类似 Javascript 中的 Promise 对象。它是一个占位符,其值会在将来被计算出来。我们可以使用

result = await future

在 future 得到值之后返回。而使用

future.set_result(xxx)

就可以设置 future 的值,也意味着 future 可以被返回了。await 操作符会自动调用 future.result() 来得到值。

loop.call_soon

通过 loop.call_soon 方法可以将一个函数插入到事件循环中。

至此,我们的异步文件读写思路也就出来了。通过 loop.call_soon 调用非阻塞读写文件的函数。若一次文件读写没有完成,则计算剩余所学读写的字节数,并再次插入事件循环直至读写完毕。

可以发现其就是把传统 Unix 编程里,非阻塞文件读写的 while 循环换成了 asyncio 的事件循环。

下面是这一过程的示意代码。

def read_step(self, future, n, total):
  res = self.fd.read(n)
  if res is None:
    self.loop.call_soon(self.read_step, future, n, total)
    return
  if not res: # EOF
    future.set_result(bytes(self.rbuffer))
    return
  self.rbuffer.extend(res)
  self.loop.call_soon(self.read_step, future, self.BLOCK_SIZE, total)

def read(self, n=-1):
  future = asyncio.Future(loop=self.loop)

  self.rbuffer.clear()
  self.loop.call_soon(self.read_step, future, min(self.BLOCK_SIZE, n), n)

  return future
Python 相关文章推荐
Python下singleton模式的实现方法
Jul 16 Python
Python常用内置函数总结
Feb 08 Python
python 出现SyntaxError: non-keyword arg after keyword arg错误解决办法
Feb 14 Python
Python实现霍夫圆和椭圆变换代码详解
Jan 12 Python
详解如何为eclipse安装合适版本的python插件pydev
Nov 04 Python
Python随机生成身份证号码及校验功能
Dec 04 Python
Python单链表原理与实现方法详解
Feb 22 Python
DataFrame 数据合并实现(merge,join,concat)
Jun 14 Python
使用keras框架cnn+ctc_loss识别不定长字符图片操作
Jun 29 Python
Python Flask异步发送邮件实现方法解析
Aug 01 Python
Django返回HTML文件的实现方法
Sep 17 Python
Python爬虫回测股票的实例讲解
Jan 22 Python
Python 如何访问外围作用域中的变量
Sep 11 #Python
Python优化技巧之利用ctypes提高执行速度
Sep 11 #Python
Python 中的with关键字使用详解
Sep 11 #Python
Python冒泡排序注意要点实例详解
Sep 09 #Python
通过5个知识点轻松搞定Python的作用域
Sep 09 #Python
python验证码识别的实例详解
Sep 09 #Python
Python随机数random模块使用指南
Sep 09 #Python
You might like
PHP通用检测函数集合
2006/11/25 PHP
php csv操作类代码
2009/12/14 PHP
解析如何通过PHP函数获取当前运行的环境 来进行判断执行逻辑(小技巧)
2013/06/25 PHP
php中error与exception的区别及应用
2014/07/28 PHP
利用php_imagick实现复古效果的方法
2016/10/18 PHP
PHP函数rtrim()使用中的怪异现象分析
2017/02/24 PHP
Redis构建分布式锁
2017/03/28 PHP
Laravel学习教程之本地化模块
2017/08/18 PHP
php使用curl伪造浏览器访问操作示例
2019/09/30 PHP
jQuery EasyUI API 中文文档 - Tree树使用介绍
2011/11/19 Javascript
学习从实践开始之jQuery插件开发 菜单插件开发
2012/05/03 Javascript
Jquery通过Ajax访问XML数据的小例子
2013/11/18 Javascript
JQuery实现当鼠标停留在某区域3秒后自动执行
2014/09/09 Javascript
javascript动态获取登录时间和在线时长
2016/02/25 Javascript
创建一个类Person的简单实例
2016/05/17 Javascript
非常酷炫的Bootstrap图片轮播动画
2016/05/27 Javascript
使用开源工具制作网页验证码的方法
2016/10/17 Javascript
Javascript DOM事件操作小结(监听鼠标点击、释放,悬停、离开等)
2017/01/20 Javascript
js数组常用最重要的方法
2018/02/04 Javascript
基于jQuery实现的设置文本区域的光标位置
2018/06/15 jQuery
详解Vue-cli3 项目在安卓低版本系统和IE上白屏问题解决
2019/04/14 Javascript
JavaScript实现简单进度条效果
2020/03/25 Javascript
Vue 401配合Vuex防止多次弹框的案例
2020/11/11 Javascript
使用PYTHON创建XML文档
2012/03/01 Python
利用ctypes提高Python的执行速度
2016/09/09 Python
Python实现针对含中文字符串的截取功能示例
2017/09/22 Python
python实现kNN算法
2017/12/20 Python
python3+PyQt5实现自定义流体混合窗口部件
2018/04/24 Python
如何在scrapy中捕获并处理各种异常
2020/09/28 Python
欧洲最大的品牌水上运动服装和设备在线零售商:Wuituit Outlet
2018/05/05 全球购物
大学生职业生涯规划书前言
2014/01/09 职场文书
测量员岗位职责
2015/02/14 职场文书
售后前台接待岗位职责
2015/04/03 职场文书
2015年库房管理工作总结
2015/10/14 职场文书
TV动画《间谍过家家》公开PV
2022/03/20 日漫
css弧边选项卡的项目实践
2023/05/07 HTML / CSS