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. 简易页面场景滚动的jquery插件

    (function($){ $.extend($.fn, { scene_scroll:function(arg_obj){ // 参数检测 // 场景数组 var $scene_arr = arg_ ...

  2. String中hashCode方法的线程安全

    class String{ //默认值是0 int hash; public int hashCode() { //将成员变量hash缓存到局部变量 int h = hash; //这里使用的是局部变 ...

  3. Could not find a package configuration file provided by 'ecl_threads' ,.................couldn't find required component 'ecl_threads'

    sudo apt-get install ros-kinetic-ecl-threads

  4. Java成神之路技术整理

    关于 Java 的技术干货,从以下几个方面汇总. Java 基础篇 Java 集合篇 Java 多线程篇 Java JVM篇 Java 进阶篇 Java 新特性篇 Java 工具篇 Java 书籍篇 ...

  5. 算法笔记--java的BigInteger类及BigDecimal类

    引包:import java.math.*; BigInteger类: 可以使用构造方法:public BigInteger(String val),或者valueOf(int)函数,如: BigIn ...

  6. Codeforces 680D - Bear and Tower of Cubes

    680D - Bear and Tower of Cubes 思路:dfs+贪心,设剩余的体积为res,存在a,使得a3 ≤ res,每次取边长为a的立方体或者边长为a-1的立方体(这时体积上限变成a ...

  7. xshell各个版本下载

    官网下载 怎么从官网下载Xshell 5 或者其他版本呢? 下面我们详细步骤说明! 1)首先我们打开netsarang官网, 点击下载Xshell 6 !填写邮箱等信息! http://www.net ...

  8. 20161226xlVBA演示文稿替换文字另存pdf

    Const ModelText As String = "机构名称" Const ModelName As String = "测试文件.pptx" Sub N ...

  9. Html显示地图

    Html可以通过JS来实现第三方地图的显示,如: 高德: 效果如下:浏览器小区域和全屏展示 代码如下:把key换成自己申请的key值 <script type="text/javasc ...

  10. CentOS 7 install Nginx

    1. rpm -Uvh http://nginx.org/packages/centos/7/noarch/RPMS/nginx-release-centos-7-0.el7.ngx.noarch.r ...