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之做一个小游戏
Sep 28 Python
Python中的defaultdict与__missing__()使用介绍
Feb 03 Python
Pandas中把dataframe转成array的方法
Apr 13 Python
Python 查找字符在字符串中的位置实例
May 02 Python
解决python中使用PYQT时中文乱码问题
Jun 17 Python
Python+AutoIt实现界面工具开发过程详解
Aug 07 Python
pandas 缺失值与空值处理的实现方法
Oct 12 Python
适合Python初学者的一些编程技巧
Feb 12 Python
Python图像处理库PIL的ImageGrab模块介绍详解
Feb 26 Python
python 6.7 编写printTable()函数表格打印(完整代码)
Mar 25 Python
30行Python代码实现高分辨率图像导航的方法
May 22 Python
Python与C++中梯度方向直方图的实现
Mar 17 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 操作excel文件的方法小结
2009/12/31 PHP
Linux中用PHP判断程序运行状态的2个方法
2014/05/04 PHP
PHP获取一年中每个星期的开始和结束日期的方法
2015/02/12 PHP
PHP不使用递归的无限级分类简单实例
2016/11/05 PHP
javascript 静态对象和构造函数的使用和公私问题
2010/03/02 Javascript
jQuery实现单行文字间歇向上滚动源代码
2013/06/02 Javascript
js 限制input只能输入数字、字母和汉字等等
2013/12/18 Javascript
javascript正则表达式参数/g与/i及/gi的使用指南
2014/08/27 Javascript
AngularJS自动表单验证
2016/02/01 Javascript
JavaScript代码实现图片循环滚动效果
2020/03/19 Javascript
js与applet相互调用的方法
2016/06/22 Javascript
微信小程序 label 组件详解及简单实例
2017/01/10 Javascript
AngularJS页面带参跳转及参数解析操作示例
2017/06/28 Javascript
Node.JS循环删除非空文件夹及子目录下的所有文件
2018/03/12 Javascript
Vue使用高德地图搭建实时公交应用功能(地图 + 附近站点+线路详情 + 输入提示+换乘详情)
2018/05/16 Javascript
layui prompt 设置允许空白提交的方法
2019/09/24 Javascript
[01:01:52]完美世界DOTA2联赛PWL S2 GXR vs Magma 第二场 11.25
2020/11/26 DOTA
在GitHub Pages上使用Pelican搭建博客的教程
2015/04/25 Python
python检测某个变量是否有定义的方法
2015/05/20 Python
Python数据类型详解(三)元祖:tuple
2016/05/08 Python
Python黑帽编程 3.4 跨越VLAN详解
2016/09/28 Python
Python基于Socket实现的简单聊天程序示例
2017/08/05 Python
对Python中range()函数和list的比较
2018/04/19 Python
对python的输出和输出格式详解
2018/12/08 Python
python中的subprocess.Popen()使用详解
2019/12/25 Python
Transpose 数组行列转置的限制方式
2020/02/11 Python
Python基于pyecharts实现关联图绘制
2020/03/27 Python
Python同时处理多个异常的方法
2020/07/28 Python
html5 worker 实例(二) 图片变换效果
2013/06/24 HTML / CSS
HTML5+CSS3模仿优酷视频截图功能示例
2017/01/05 HTML / CSS
学生安全责任书
2014/04/15 职场文书
关于责任的演讲稿
2014/05/20 职场文书
孝老爱亲模范事迹材料
2014/05/25 职场文书
班主任寄语2016
2015/12/04 职场文书
导游词之新疆尼雅遗址
2019/10/16 职场文书
vue+elementui 实现新增和修改共用一个弹框的完整代码
2021/06/08 Vue.js