Python开发的高级技巧

甲壳虫 7年前
   <p>本文我列出几个Python高级技巧:</p>    <p>1. contextmanager</p>    <p>写Python代码的时候经常将一系列操作放在一个语句块中,Python 2.5加入了with语法,实现上下文管理功能,这让代码的可读性更强并且错误更少。最常见的例子就是open,如果不使用with,使用open会是这样:</p>    <p><img src="https://simg.open-open.com/show/5a801cfae9e481a8f951b38c9b8b7aef.jpg"></p>    <p>如果使用with,可以简化为两行:</p>    <p><img src="https://simg.open-open.com/show/17a19486b9f5864a8e3665aeb47638c7.jpg"></p>    <p>在执行完缩进的代码块后会自动关闭文件。创建上下文管理器实际就是创建一个类,添加__enter__和__exit__方法。看看如何实现open的上下文管理功能:</p>    <p><img src="https://simg.open-open.com/show/e3f880cefdfe34e70f015416fe4a6362.jpg"></p>    <p>自定义上下文管理器确实很方便,但是Python标准库还提供了更易用的上下文管理器工具模块contextlib,它是通过生成器实现的,我们不必再创建类以及__enter__和__exit__这两个特殊的方法:</p>    <p><img src="https://simg.open-open.com/show/e98dd34696a4ef0a21aa329ced307a55.jpg"></p>    <p>yield关键词把上下文分割成两部分:yield之前就是__init__中的代码块;yield之后其实就是__exit__中的代码块;yield生成的值会绑定到with语句as子句中的变量(如果没有生成,也就没有as字句)。</p>    <p>2. total_ordering 。对比自定义对象需要添加__lt__、__le__、__gt__、__ge__和__eq__等方法,如果使用total_ordering,只需要定义__eq__以及__lt__、__le__、__gt__、__ge__四种之一就可以了:</p>    <p><img src="https://simg.open-open.com/show/c2a3f933c06f3110e5fe0ef81956fb2e.jpg"></p>    <p>3. 有时候BUG隐藏的太深,需要对上下文都有清晰的展示来帮助判断。用pdb调试不方便,用print不直观。可以使用如下函数获取当前调用栈:</p>    <p><img src="https://simg.open-open.com/show/0f16abf715ed1163bded330599f4234d.jpg"></p>    <p>4. inspect 。有时候我们想查看一下对象的一些信息或者做类型检查,也就是自省(检查某些事物以确定它是什么、它知道什么以及它能做什么):</p>    <p><img src="https://simg.open-open.com/show/6c7caf5cfbcc513cc6d6f6cbd0c7ea8c.jpg"></p>    <p>它在实际工作中还能有什么意义,通过自省获取方法的参数,从而设置缓存的键,如 flask-cache( https://github.com/thadeusb/flask-cache/blob/master/flask_cache/__init__.py#L418)和 douban-mc(https://github.com/douban/douban-mc/blob/master/douban/mc/decorator.py#L39)</p>    <p>5. Mixin模式 。它是什么先看「Mixin是什么概念? 」 :</p>    <p>假如我们想通过python内置类型Dict的方式来存放数据,编写的类可以混入DictMixin就好了:</p>    <p><img src="https://simg.open-open.com/show/0bd401f95710606b09de49170e7b7907.jpg"></p>    <p>PS: 如果你想兼容Python 2.6以下和Python 3,可以使用collections.MutableMapping:</p>    <p><img src="https://simg.open-open.com/show/83aa59e58913289fb54b6abcc5df78bd.jpg"></p>    <p>但是MutableMapping需要额外实现__iter__和__len__。</p>    <p>PPS: MutableMapping是学习实现抽象类的范例:</p>    <p> </p>    <p>它继承了Iterable和Sized,而Iterable中通过abstractmethod要求你必须定义__iter__方法,Sized中要求你必须定义__len__方法,否则就会提示:</p>    <p>TypeError: Can't instantiate abstract class MyDict with abstract metho</p>    <p>ds __iter__, __len__</p>    <p> </p>    <p>来自:http://developer.51cto.com/art/201611/522852.htm</p>    <p> </p>