Effective Python

第1章 用Pythonic方式来思考

  • be pythonic

  • 遵守pep8

  • python3有两种字符序列类型:bytes(原始的字节)和str(Unicode字符).

  • 在python3中需要用二进制方式读写文件时,要用wb和rb

  • 应该尽可能使用if/else表达式和辅助函数来使代码清晰

  • 不要在单次切片中同时指定start, end和stride.可以采用两步进行范围切割和步进切割.

  • 使用列表推导来代替map和filter

  • 不要在列表推导中使用两个以上的表达式.使用if和for语句代替.

  • 如果数据量很大,用生成器表达式来改写列表推导.生成器表达式会返回一个迭代器,然后可以用next来生成新值.

  • 如果需要同时使用list的索引和元素,最好使用enumerate替代range

  • 用zip函数同时遍历2个迭代器

  • 利用try/except/else/finally进行异常处理

第2章 函数

  • 尽量用异常来表示特殊情况,而不是None

  • python支持闭包,闭包是一种定义在某个作用域的函数,这种函数引用了那个作用域里的变量.

  • python的函数是一级对象,也就是可以直接引用,可以赋给变量,可以当成参数传给其他函数.

  • 函数中的局部变量不会影响外部作用域.如果需要获取闭包函数内的数据,应该使用nonlocal声明变量.如果使用nonlocal让代码变得复杂,应该将相关的函数封装成辅助类.

  • 如果函数返回列表,可以考虑用生成器改写,将return变为yield.

  • 如果需要在函数中多次迭代一个生成器,可以考虑把写一个容器,把__iter__方法实现为生成器.

  • 令函数接受可选的位置参数*args,能够使代码清晰.如果要把列表中的所有元素作为位置参数传入,就在列表前加上*.

  • 可选参数有两个问题,一个是过长的生成器作为参数会导致消耗大量内存,一个是不方便修改函数.

  • 可以用关键字参数表示函数可选的行为和默认值.参数的默认值会在模块加载的时候求出.因此不要在参数默认值中创建[]或{}等动态的值.解决方法是把默认值设为None,并在docstring中具体解释.

  • 可以在定义函数的时候用*来分隔位置参数和关键字参数(只能以关键字指定的参数)

第3章 类与继承

  • 当数据结构比较复杂时,尽量用辅助类来维护程序的状态,而不要用字典和元组.

  • 类似sort的key参数可以接受一个函数,这称挂钩(hook)函数.

  • 简单的接口应该接受函数而不是类的实例,因为函数是一级对象.

  • 如果要用函数保存状态,应该定义新的类并实现__call__方法

  • 通过@classmethod可以构造多态的类的对象.

  • 较复杂的继承体系中,如果仍让子类直接调用父类的__init__,会出现很多意想不到的行为,因此建议用super初始化父类.

  • mix-iin是一种只定义了其他类可能需要提供的一套附加方法的小型类.只在使用mix-in组件类时进行多重继承.

  • Python类的属性只有public和private.一般还会用单个下划线开头来习惯性命名protected.

  • private属性是开头至少有两个下划线,结尾至多有一个下划线的属性.python对于__private_name的保存机制实际上是_ClassName__private_name.因此可以通过这种方式访问所有private属性.

  • 应该多用前者,少用后者.

  • 如果要定制简单子类,可以直接从python的容器类型继承.否则可以继承collections.abc中的抽象基类以实现自定义的容器类型.

第4章 元类及属性

  • 用纯属性取代get和set方法

  • 如果想要使用get和set,应该使用@property修饰器的getter和setter.它的缺点是不便于复用.

  • 描述符可以把同一套逻辑运用在类中的不同属性上面.

  • WeakKeyDictionary可以保证描述符类不会泄露内存

第5章 并发及并行

  • 并发(concurrency)是指计算机通过迅速切换看似在同一时间做很多不同的事.并行(parallelism)则是指计算机确实在同一时间做很多不同的事.

  • subprocess用于与外部进程交互.

  • Python解释器能平行地运行多条子进程.

  • GIL确保了字节码解释器的协调性,但也带来了同一时刻只能进行一条线程的负面影响.因此Python的多线程适用于处理阻塞式的I/O操作,而不是平行计算.

  • Python多线程同样需要使用Lock来防止数据竞争,因为解释器可能会在线程的任意时刻暂停它们.

  • 用queue.QUeue来协调各线程之间的工作.它解决了并发式管道存在的问题.

  • 线程存在的问题是:代码复杂,需要占用大量内存(大约8mb),启动时的开销比较大.协程可以避免这些问题.

  • 协程的工作原理是:每当生成器函数执行到yield表达式时,消耗生成器的那段代码,就通过send方法给生成器回传一个值.

  • concurrent.futures会以子进程的形式平行地运行多个解释器,从而令python能够利用多核心CPU来提升速度.这种做法适用于较为孤立且数据利用率高的任务.

第6章 内置模块

  • functools.wraps可以定义函数修饰器,用于在函数执行之前以及执行完毕之后分别运行一些附加代码.可以实现约束语义,调试程序,注册函数等目标.

  • contextlib模块可以使自己编写的对象和函数支持with语句.

  • python的pickle模块能将python对象序列化为字节流,也能将这些字节反序列化为python对象.但是这种序列化数据采用的是不安全的格式,如果其中有恶意信息,在反序列化时可能对程序造成损害.Json数据则是安全的.

  • copyreg能使pickle变得可靠.

  • 使用datetime而不是time来处理本地时间

  • 使用内置的算法和数据结构,例如collections.deque双向队列,collections.OrderedDict有序字典,collections.defaultdict带有默认值的字典,heapq堆,bisect二分查找,itertools处理迭代器.

  • 在重视精确度的场合,应该使用decimal

第7章 协作开发

  • 为每个函数,类和模块编写docstring.

  • 用包来安排模块,并提供稳固的API.

  • 为自编的模块定义根异常,以便将调用者与API相隔离.

第8章 部署

  • 可以用模块级别的代码来配置不同的部署环境

  • 通过repr来输出调试信息

  • 用unittest来测试代码

  • pdb模块的set_trace()可以让程序在执行到某一行的时候暂停,用于调试.

  • 利用cProfile来分析每个模块的性能,然后再优化.

  • 用tarcemalloc来掌握内存的使用及泄露情况.

《Effective Python:编写高质量Python代码的59个有效方法》读书笔记(完结)的更多相关文章

  1. Effective Python 编写高质量Python代码的59个有效方法

    Effective Python 编写高质量Python代码的59个有效方法

  2. 编写高质量JavaScript代码的68个有效方法

    简介: <Effective JavaScript:编写高质量JavaScript代码的68个有效方法>共分为7章,分别涵盖JavaScript的不同主题.第1章主要讲述最基本的主题,如版 ...

  3. 编写高质量JS代码的68个有效方法(八)

    [20141227]编写高质量JS代码的68个有效方法(八) *:first-child { margin-top: 0 !important; } body>*:last-child { ma ...

  4. 编写高质量JS代码的68个有效方法(七)

    [20141220]编写高质量JS代码的68个有效方法(七) *:first-child { margin-top: 0 !important; } body>*:last-child { ma ...

  5. 编写高质量JS代码的68个有效方法(六)

    [20141213]编写高质量JS代码的68个有效方法(六) *:first-child { margin-top: 0 !important; } body>*:last-child { ma ...

  6. 编写高质量JS代码的68个有效方法(四)

    [20141129]编写高质量JS代码的68个有效方法(四) *:first-child { margin-top: 0 !important; } body>*:last-child { ma ...

  7. 编写高质量JS代码的68个有效方法(三)

    [20141030]编写高质量JS代码的68个有效方法(三) *:first-child { margin-top: 0 !important; } body>*:last-child { ma ...

  8. 编写高质量JS代码的68个有效方法(二)

    [20141011]编写高质量JS代码的68个有效方法(二) *:first-child { margin-top: 0 !important; } body>*:last-child { ma ...

  9. JavaScript手札:《编写高质量JS代码的68个有效方法》(一)(1~5)

    编写高质量JS代码的68个有效方法(一) *:first-child { margin-top: 0 !important; } body>*:last-child { margin-botto ...

  10. 编写高质量JS代码的68个有效方法(十三)

    No.61.不要阻塞I/O事件队列 Tips: 异步API使用回调函数来延缓处理代价高昂的操作以避免阻塞主应用程序 JavaScript并发的接收事件,但会使用一个事件队列按序地处理事件处理程序 在应 ...

随机推荐

  1. python 将列表中的字符串转为数字

    本文实例讲述了Python中列表元素转为数字的方法.分享给大家供大家参考,具体如下: 有一个数字字符的列表: numbers = ['1', '5', '10', '8'] 想要把每个元素转换为数字: ...

  2. 未能加载文件或程序集“Newtonsoft.Json, Version=4.5.0.0

    错误描述: 错误原因: 因为引用出了问题,在你的程序集里面找不到的Newtonsoft.Json,所以它就拿从系统盘里面预装的旧版的来用,结果就报版本错误了. 解决方案: web.config  的  ...

  3. Activity生命周期之我见

    关于Activity生命周期的文章很多,而且大部分也说得很详细,所以作为关于这方面的内容我本来不想多说,但是大家可能跟我之前一样,在看这方面的内容的时候都能很容易地看懂,但是过几天又忘了,或者在用的程 ...

  4. ubuntu下xx-net可用IP为0, ping6显示 Network is unreachable

    不知和xx-net有没有关系. 全新安装系统 sudo apt-get install miredo 这时, ping6 ipv6.google.com 是能ping通的 按github上依次安装py ...

  5. java后台校验 hibernate validator

    链接 : https://www.cnblogs.com/softidea/p/6044123.html

  6. Codeforces 260D - Black and White Tree

    260D - Black and White Tree 思路:把两种颜色先按值sort一下,最小值肯定是叶子,然后把这个叶子和另外一中颜色的一个最小值的节点连接起来,再把这个节点变成叶子,把值减掉就可 ...

  7. 多年未写过java了

    java面试必须了解和记忆的知识点(linux,nginx,数据库之类的暂时不考虑了) java面试的115常见问题,这个博客还是不过的推荐一下吧 http://www.importnew.com/1 ...

  8. [Java学习] java泛型通配符和类型参数的范围

    本节先讲解如何限制类型参数的范围,再讲解通配符(?). 类型参数的范围 在泛型中,如果不对类型参数加以限制,它就可以接受任意的数据类型,只要它是被定义过的.但是,很多时候我们只需要一部分数据类型就够了 ...

  9. 20170706wdVBA保存图片到本地API

    Private Type GUID Data1 As Long Data2 As Integer Data3 As Integer Data4(0 To 7) As Byte End Type Pri ...

  10. spfa毒瘤算法

    终于知道怎么卡spfa(不优化)这一毒瘤算法了 下面就是造数据代码,点数才1e5,边数379980 随便测了一组数据: count: 831841219(入队次数) 68917.096 ms(足够t到 ...