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编写基于DHT协议的BT资源爬虫
Mar 19 Python
python GUI实例学习
Nov 21 Python
Python实现的选择排序算法原理与用法实例分析
Nov 22 Python
Python多线程爬虫实战_爬取糗事百科段子的实例
Dec 15 Python
Python使用Scrapy爬虫框架全站爬取图片并保存本地的实现代码
Mar 04 Python
在python中实现对list求和及求积
Nov 14 Python
Tensorflow分类器项目自定义数据读入的实现
Feb 05 Python
python如何将两个txt文件内容合并
Oct 18 Python
pyinstaller打包单文件时--uac-admin选项不起作用怎么办
Apr 15 Python
python在一个范围内取随机数的简单实例
Aug 16 Python
Ubuntu权限不足无法创建文件夹解决方案
Nov 14 Python
Python 详解通过Scrapy框架实现爬取百度新冠疫情数据流程
Nov 11 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格式化电话号码的方法
2015/04/24 PHP
Yii多表联合查询操作详解
2016/06/02 PHP
Javascript的IE和Firefox兼容性汇编(zz)
2007/02/02 Javascript
JavaScript 类型的包装对象(Typed Wrappers)
2011/10/27 Javascript
JavaScript中常用的运算符小结
2012/01/18 Javascript
详解如何较好的使用js
2016/12/16 Javascript
js实现网页定位导航功能
2017/03/07 Javascript
干货!教大家如何选择Vue和React
2017/03/13 Javascript
详解JS中的attribute属性
2017/04/25 Javascript
jQuery输入框密码的显示隐藏【代码分享】
2017/04/29 jQuery
angularjs2中父子组件的数据传递的实例代码
2017/07/05 Javascript
红黑树的插入详解及Javascript实现方法示例
2018/03/26 Javascript
使用vue打包时vendor文件过大或者是app.js文件很大的问题
2018/06/29 Javascript
Vue中全局变量的定义和使用
2019/06/05 Javascript
监控Nodejs的性能实例代码
2019/07/02 NodeJs
小程序实现简单语音聊天的示例代码
2020/07/24 Javascript
[01:33]真香警告!DOTA2勇士令状不朽珍藏Ⅱ饰品欣赏
2018/06/26 DOTA
python 线程的暂停, 恢复, 退出详解及实例
2016/12/06 Python
socket + select 完成伪并发操作的实例
2017/08/15 Python
Python基于回溯法子集树模板解决马踏棋盘问题示例
2017/09/11 Python
Django中的文件的上传的几种方式
2018/07/23 Python
详解将Django部署到Centos7全攻略
2018/09/26 Python
python实现给scatter设置颜色渐变条colorbar的方法
2018/12/13 Python
关于ResNeXt网络的pytorch实现
2020/01/14 Python
Python把图片转化为pdf代码实例
2020/07/28 Python
Python3.9.0 a1安装pygame出错解决全过程(小结)
2021/02/02 Python
国贸专业自荐信范文
2014/03/02 职场文书
技校毕业生自荐信范文
2014/03/07 职场文书
房产买卖委托公证书
2014/04/04 职场文书
爱心活动计划书
2014/04/26 职场文书
技术员个人工作总结
2015/03/03 职场文书
美容院管理规章制度
2015/08/05 职场文书
导游词之永泰公主墓
2019/12/04 职场文书
mysql 8.0.24 安装配置方法图文教程
2021/05/12 MySQL
Qt自定义Plot实现曲线绘制的详细过程
2021/11/02 Python
JavaScript 中for/of,for/in 的详细介绍
2021/11/17 Javascript