Python Cookbook 3rd Edition Documentation

文章目录

  • 第一章:数据结构和算法
    • 1.1 解压序列赋值给多个变量
      • 问题
      • 解决方案
      • 讨论
    • 1.2 解压可迭代对象赋值给多个变量
      • 问题
      • 解决方案
      • 讨论

第一章:数据结构和算法

Python 提供了大量的内置数据结构,包括列表,集合以及字典。大多数情况下使用这些数据结构是很简单的。 但是,我们也会经常碰到到诸如查询,排序和过滤等等这些普遍存在的问题。 因此,这一章的目的就是讨论这些比较常见的问题和算法。

1.1 解压序列赋值给多个变量

问题

现在有一个包含 N 个元素的元组或者是序列,怎样将它里面的值解压后同时赋值给 N 个变量?

解决方案

任何的序列(或者是可迭代对象)可以通过一个简单的赋值语句解压并赋值给多个变量。 唯一的前提就是变量的数量必须跟序列元素的数量是一样的。

代码示例:

>>> p = (4, 5)
>>> x, y = p
>>> x
4
>>> y
5
>>>
>>> data = [ 'ACME', 50, 91.1, (2012, 12, 21) ]
>>> name, shares, price, date = data
>>> name
'ACME'
>>> date
(2012, 12, 21)
>>> name, shares, price, (year, mon, day) = data
>>> name
'ACME'
>>> year
2012
>>> mon
12
>>> day
21
>>>

如果变量个数和序列元素的个数不匹配,会产生一个异常。

讨论

实际上,这种解压赋值可以用在任何可迭代对象上面,而不仅仅是列表或者元组。 包括字符串,文件对象,迭代器和生成器。

代码示例:

>>> s = 'Hello'
>>> a, b, c, d, e = s
>>> a
'H'
>>> b
'e'
>>> e
'o'
>>>

有时候,你可能只想解压一部分,丢弃其他的值。对于这种情况 Python 并没有提供特殊的语法。 但是你可以使用任意变量名去占位,到时候丢掉这些变量就行了。

代码示例:

>>> data = [ 'ACME', 50, 91.1, (2012, 12, 21) ]
>>> _, shares, price, _ = data
>>> shares
50
>>> price
91.1
>>>

你必须保证你选用的那些占位变量名在其他地方没被使用到。

1.2 解压可迭代对象赋值给多个变量

问题

如果一个可迭代对象的元素个数超过变量个数时,会抛出一个 ValueError 。 那么怎样才能从这个可迭代对象中解压出 N 个元素出来?

解决方案

Python 的星号表达式可以用来解决这个问题。比如,你在学习一门课程,在学期末的时候, 你想统计下家庭作业的平均成绩,但是需要排除掉第一个和最后一个分数。如果只有四个分数,你可能就直接去简单的手动赋值, 但如果有 24 个呢?这时候星号表达式就派上用场了:

def drop_first_last(grades):first, *middle, last = gradesreturn avg(middle)

另外一种情况,假设你现在有一些用户的记录列表,每条记录包含一个名字、邮件,接着就是不确定数量的电话号码。 你可以像下面这样分解这些记录:

>>> name, email, *phone_numbers = record
>>> name
'Dave'
>>> email
'dave@example.com'
>>> phone_numbers
['773-555-1212', '847-555-1212']
>>>

值得注意的是上面解压出的 phone_numbers 变量永远都是列表类型,不管解压的电话号码数量是多少(包括 0 个)。 所以,任何使用到 phone_numbers 变量的代码就不需要做多余的类型检查去确认它是否是列表类型了。

星号表达式也能用在列表的开始部分。比如,你有一个公司前 8 个月销售数据的序列, 但是你想看下最近一个月数据和前面 7 个月的平均值的对比。你可以这样做:

*trailing_qtrs, current_qtr = sales_record
trailing_avg = sum(trailing_qtrs) / len(trailing_qtrs)
return avg_comparison(trailing_avg, current_qtr)

讨论

扩展的迭代解压语法是专门为解压不确定个数或任意个数元素的可迭代对象而设计的。 通常,这些可迭代对象的元素结构有确定的规则(比如第 1 个元素后面都是电话号码), 星号表达式让开发人员可以很容易的利用这些规则来解压出元素来。 而不是通过一些比较复杂的手段去获取这些关联的元素值。

值得注意的是,星号表达式在迭代元素为可变长元组的序列时是很有用的。 比如,下面是一个带有标签的元组序列:

records = [('foo', 1, 2),('bar', 'hello'),('foo', 3, 4),
]def do_foo(x, y):print('foo', x, y)def do_bar(s):print('bar', s)for tag, *args in records:if tag == 'foo':do_foo(*args)elif tag == 'bar':do_bar(*args)

星号解压语法在字符串操作的时候也会很有用,比如字符串的分割

代码示例:

>>> line = 'nobody:*:-2:-2:Unprivileged User:/var/empty:/usr/bin/false'
>>> uname, *fields, homedir, sh = line.split(':')
>>> uname
'nobody'
>>> homedir
'/var/empty'
>>> sh
'/usr/bin/false'
>>>

有时候,你想解压一些元素后丢弃它们,你不能简单就使用 * , 但是你可以使用一个普通的废弃名称,比如 _ 或者 ign (ignore)。

代码示例:

>>> record = ('ACME', 50, 123.45, (12, 18, 2012))
>>> name, *_, (*_, year) = record
>>> name
'ACME'
>>> year
2012
>>>

在很多函数式语言中,星号解压语法列表处理有许多相似之处。比如,如果你有一个列表, 你可以很容易的将它分割成前后两部分:

>>> items = [1, 10, 7, 4, 5, 9]
>>> head, *tail = items
>>> head
1
>>> tail
[10, 7, 4, 5, 9]
>>>

如果你够聪明的话,还能用这种分割语法去巧妙的实现递归算法。比如:

>>> def sum(items):
...     head, *tail = items
...     return head + sum(tail) if tail else head
...
>>> sum(items)
36
>>>

然后,由于语言层面的限制,递归并不是 Python 擅长的。 因此,最后那个递归演示仅仅是个好奇的探索罢了,对这个不要太认真了。

Python Cookbook 3rd Edition Documentation相关推荐

  1. Oreilly.Python.Cookbook(3rd.Edition.May.2013)pdf

    下载地址:网盘下载 转载于:https://www.cnblogs.com/long12365/p/9730131.html

  2. 《Python Cookbook 3rd》笔记(1.4):查找最大或最小的N个元素

    <Python Cookbook 3rd>1.4:查找最大或最小的N个元素 问题 怎样从一个集合中获得最大或者最小的N个元素列表? 解法 heapq 模块有两个函数:nlargest()和 ...

  3. 《Python Cookbook 3rd》笔记(5.21):序列化 Python 对象

    序列化 Python 对象 问题 你需要将一个 Python 对象序列化为一个字节流,以便将它保存到一个文件.存储到数据库或者通过网络传输它. 解法 对于序列化最普遍的做法就是使用 pickle 模块 ...

  4. 《Python Cookbook 3rd》笔记(5.20):与串行端口的数据通信

    与串行端口的数据通信 问题 你想通过串行端口读写数据,典型场景就是和一些硬件设备打交道 (比如一个机器人或传感器). 解法 尽管你可以通过使用 Python 内置的 I/O 模块来完成这个任务,但对于 ...

  5. 《Python Cookbook 3rd》笔记(5.18):将文件描述符包装成文件对象

    将文件描述符包装成文件对象 问题 你有一个对应于操作系统上一个已打开的 I/O 通道 (比如文件.管道.套接字等)的整型文件描述符,你想将它包装成一个更高层的 Python 文件对象. 解法 一个文件 ...

  6. 《Python Cookbook 3rd》笔记(5.16):增加或改变已打开文件的编码

    增加或改变已打开文件的编码 问题 你想在不关闭一个已打开的文件前提下增加或改变它的 Unicode 编码. 解法 如果你想给一个以二进制模式打开的文件添加 Unicode 编码/解码方式,可以使用io ...

  7. 《Python Cookbook 3rd》笔记(5.15):打印不合法的文件名

    打印不合法的文件名 问题 你的程序获取了一个目录中的文件名列表,但是当它试着去打印文件名的时候程序崩溃,出现了 UnicodeEncodeError 异常和一条奇怪的消息-- surrogates n ...

  8. 《Python Cookbook 3rd》笔记(5.14):忽略文件名编码

    忽略文件名编码 问题 你想使用原始文件名执行文件的 I/O 操作,也就是说文件名并没有经过系统默认编码去解码或编码过. 解法 默认情况下,所有的文件名都会根据 sys.getfilesystemenc ...

  9. 《Python Cookbook 3rd》笔记(5.10):内存映射的二进制文件

    内存映射的二进制文件 问题 你想内存映射一个二进制文件到一个可变字节数组中,目的可能是为了随机访问它的内容或者是原地做些修改. 解法 使用 mmap 模块来内存映射文件.下面是一个工具函数,向你演示了 ...

最新文章

  1. 第三方医药物流的现状及发展
  2. 机房收费--上机状态查询
  3. 网易云音乐:基于分布式图学习PGL的推荐系统优化之路
  4. 微软开源AI诊断工具Error Analysis
  5. 易语言程序应用程序错误退出_为什么我退出Google并构建了一个向孩子们教授个人理财的应用程序
  6. ios textView显示不全
  7. 转贴 30岁你会站在哪?
  8. cherry 键盘失灵记录
  9. NVIDIA GeForce Experience 无法登录解决办法
  10. [搜索 meet in the middle+哈希] ProjectEuler 598. Split Divisibilities
  11. IoT当前最重要的机遇,全面解读专为边缘计算而生的EdgeX Foundry
  12. BUUCTF MISC刷题笔记(三)
  13. 慢聊Go之Go常见的Web 开发框架
  14. 系统设计面试的万金油
  15. 怎样把软件从计算机移到桌面上,怎么把电脑软件移动到D盘
  16. 工赋开发者社区 | MES与ERP/APS/PLM等的系统集成技术
  17. 品牌统一化vs多元化:美团与滴滴为何选择了不同道路?
  18. 用于自动驾驶的混合稀疏-稠密单目SLAM系统
  19. Dalvik字节码类型对照表
  20. VScode注释中文时乱码怎么办?VScode中文注释乱码问题的解决方法

热门文章

  1. PAT乙类1013 数素数 (20 分)
  2. 一份北大信科内部流传的 “CS 自救指南”
  3. Jarvis:一个值得关注的多模态端到端人机对话框架,针对所有行业适配
  4. 解开玻尔兹曼机的封印会发生什么?
  5. Java远程通讯技术及原理分析
  6. windows下安装Sublime+Miktex+Sumatrapdf配置LATEX环境
  7. 剑指offer-python解答
  8. 国科金:共融机器人基础理论与关键技术研究重大研究计划
  9. C++ 类设计核查表
  10. bzoj 1596 电话网络