Python 开发的高级技巧
AlfPalafox
8年前
<p>本文我列出几个Python高级技巧:</p> <p>1. contextmanager<br> 写Python代码的时候经常将一系列操作放在一个语句块中,Python 2.5加入了with语法,实现上下文管理功能,这让代码的可读性更强并且错误更少。最常见的例子就是open,如果不使用with,使用open会是这样:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/c93a92799dd9be1a25162381bbb48355.jpg"></p> <p>如果使用with,可以简化为两行:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/768f2f0da20fac546de5586552dca351.jpg"></p> <p>在执行完缩进的代码块后会自动关闭文件。创建上下文管理器实际就是创建一个类,添加__enter__和__exit__方法。看看如何实现open的上下文管理功能:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/1fc6dfe5d7d1023005180dd7db3faa79.jpg"></p> <p>自定义上下文管理器确实很方便,但是Python标准库还提供了更易用的上下文管理器工具模块contextlib,它是通过生成器实现的,我们不必再创建类以及__enter__和__exit__这两个特殊的方法:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/b9b4deb2cac4ae57fda2ff85312b0191.jpg"></p> <p>yield关键词把上下文分割成两部分:yield之前就是__init__中的代码块;yield之后其实就是__exit__中的代码块;yield生成的值会绑定到with语句as子句中的变量(如果没有生成,也就没有as字句)。</p> <p>2. <strong>total_ordering</strong> 。对比自定义对象需要添加__lt__、__le__、__gt__、__ge__和__eq__等方法,如果使用total_ordering,只需要定义__eq__以及__lt__、__le__、__gt__、__ge__四种之一就可以了:</p> <p style="text-align:center"><img alt="Python 开发的高级技巧" src="https://simg.open-open.com/show/2ac8827236ed2e69fb01cf96c9316e50.jpg"></p> <p>3. 有时候BUG隐藏的太深,需要对上下文都有清晰的展示来帮助判断。用pdb调试不方便,用print不直观。可以使用如下函数获取当前调用栈:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/cde415d3e811a25e1216c3bb81dc9c02.jpg"></p> <p>4. <strong>inspect</strong> 。有时候我们想查看一下对象的一些信息或者做类型检查,也就是自省(检查某些事物以确定它是什么、它知道什么以及它能做什么):</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/7f1d55005c9e180a2d22daecfe54c5fd.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><br> <strong>5. Mixin模式</strong> 。它是什么先看「Mixin是什么概念? 」 :</p> <p>假如我们想通过python内置类型Dict的方式来存放数据,编写的类可以混入DictMixin就好了:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/f12c4c5f126719a7591b5b6ffd3bc76f.jpg"></p> <p>PS: 如果你想兼容Python 2.6以下和Python 3,可以使用collections.MutableMapping:</p> <p style="text-align:center"><img src="https://simg.open-open.com/show/414b8e3d3851dbb940434e34bbaf7420.jpg"></p> <p>但是MutableMapping需要额外实现__iter__和__len__。</p> <p>它继承了Iterable和Sized,而Iterable中通过abstractmethod要求你必须定义__iter__方法,Sized中要求你必须定义__len__方法,否则就会提示:<br> </p> <p>TypeError: Can't instantiate abstract class MyDict with abstract metho<br> ds __iter__, __len__</p> <p> </p> <p>来自:http://mp.weixin.qq.com/s?__biz=MzA3NDk1NjI0OQ==&mid=2247483770&idx=1&sn=a75a70d0d49f4378818fec89ea6631a0&chksm=9f76adcca80124dae4ed5105a9d9938f381cebeaf682ae103dce38c1720f41426c7b54bd90da</p> <p> </p>