- 1.A 为什么一定要掌握自学能力?
- 1.B 为什么把编程当作自学的入口?
- 1.C 只靠阅读习得新技能
- 1.D 开始阅读前的一些准备
- 1.E.1 入口
- 1.E.2 值及其相应的运算
- 1.E.3 流程控制
- 1.E.4 函数
- 1.E.5 字符串
- 1.E.6 数据容器
- 1.E.7 文件
- 1.F 如何从容应对含有过多 “过早引用” 的知识?
- 1.G 官方教程:The Python Tutorial
- 2.A 笨拙与耐心
- 2.B 刻意练习
- 2.C 为什么从函数开始?
- 2.D.1 关于参数(上)
- 2.D.2 关于参数(下)
- 2.D.3 化名与匿名
- 2.D.4 递归函数
- 2.D.5 函数的文档
- 2.D.6 保存到文件的函数
- 2.D.7 测试驱动的开发
- 2.D.8 可执行的 Python 文件
- 2.E 刻意思考
- 3.A 战胜难点
- 3.B.1 类 —— 面向对象编程
- 3.B.2 类 —— Python 的实现
- 3.B.3 函数工具
- 3.B.4 正则表达式
- 3.B.5 BNF 以及 EBNF
- 3.C 拆解
- 3.D 刚需幻觉
- 3.E 全面 —— 自学的境界
- 3.F 自学者的社交
- 3.G 这是自学者的黄金时代
- 3.H 避免注意力漂移
化名与匿名
化名
在 Python 中,我们可以给一个函数取个化名(alias)。
以下的代码,我们先是定义了一个名为 _is_leap
的函数,而后为它另取了一个化名:
def _is_leap(year):
return year % 4 == 0 and (year % 100 != 0 or year % 400 == 0)
year_leap_bool = _is_leap
print(year_leap_bool) #<function __main__._is_leap(year)>
print(year_leap_bool(800)) # _is_leap(800) -> True
print(id(year_leap_bool)) # id() 这个函数可以查询某对象的内存地址
print(id(_is_leap)) # year_leap_bool 和 _is_leap 其实保存在同一个地址中,也就是说,它们是同一个对象。
print(type(year_leap_bool))
print(type(_is_leap)) # 它们都是 function
<function __main__._is_leap(year)>
True
4547071648
4547071648
function
function
我们可以看到的是,id(year_leap_bool)
和 id(_is_leap)
的内存地址是一样的 —— 它们是同一个对象,它们都是函数。所以,当你写 year_leap_bool = _is_leap
的时候,相当于给 _is_leap()
这个函数取了个化名。
在什么样的情况下,要给一个函数取一个化名呢?
在任何一个工程里,为函数或者变量取名都是很简单却不容易的事情 —— 因为可能会重名(虽然已经尽量用变量的作用域隔离了),可能会因取名含混而令后来者费解……
所以,仅仅为了少敲几下键盘而给一个函数取个更短的化名,实际上并不是好主意,更不是好习惯。尤其现在的编辑器都支持自动补全和多光标编辑的功能,变量名再长都不构成负担。
更多的时候,为函数取一个化名,应该是为了提高代码可读性 —— 对自己或其他人都很重要。
lambda
写一个很短的函数可以用 lambda
关键字。
下面是用 def
关键字写函数:
def add(x, y):
return x + y
print(add(3, 5))
8
下面是用 lambda
关键字写函数:
add = lambda x, y: x + y
print(add(3, 5))
8
lambda 的语法结构如下:
lambda_expr ::= "lambda" [parameter_list] ":" expression
以上使用的是 BNF 标注。当然,BNF 是你目前并不熟悉的,所以,有疑惑别当回事。
反正你已经见到示例了:
lambda x, y: x + y
先写上 lambda
这个关键字,其后分为两个部分,:
之前是参数,之后是表达式;这个表达式的值,就是这个函数的返回值。
注意:
lambda
语句中,:
之后有且只能有一个表达式。
而这个函数呢,没有名字,所以被称为 “匿名函数”。
add = lambda x, y: x + y
就相当于是给一个没有名字的函数取了个名字。
lambda 的使用场景
那 lambda 这种匿名函数的用处在哪里呢?
作为某函数的返回值
第一个常见的用处是作为另外一个函数的返回值。
让我们看看 The Python Tutorial 中的一个例子。
def make_incrementor(n):
return lambda x: x + n
f = make_incrementor(42)
print(f(0))
print(f(1))
42
43
这个例子乍看起来很令人迷惑。我们先看看 f = make_incrementor(42)
之后,f
究竟是什么东西:
def make_incrementor(n):
return lambda x: x + n
f = make_incrementor(42)
print(f)
print(id(make_incrementor))
print(id(f))
<function __main__.make_incrementor.<locals>.<lambda>(x)>
4428443296
4428726888
首先,要注意,f
并不是 make_incrementor()
这个函数的化名,如果是给这个函数取个化名,写法应该是:
f = make_incrementor
那 f
是什么呢?它是 <function __main__.make_incrementor.<locals>.<lambda>(x)>
:
f = make_incrementor(42)
是将make_incrementor(42)
的返回值保存到f
这个变量之中;- 而
make_incrementor()
这个函数接收到42
这个参数之后,返回了一个函数:lambda x: x + 42
;- 于是,
f
中保存的函数是lambda x: x + 42
;- 所以,
f(0)
是向这个匿名函数传递了0
,而后,它返回的是0 + 42
。
作为某函数的参数
可以拿一些可以接收函数为参数的内建函数做例子。比如,map()
。
map
(function, iterable, ...)Return an iterator that applies function to every item of iterable, yielding the results. If additional iterable arguments are passed, function must take that many arguments and is applied to the items from all iterables in parallel. With multiple iterables, the iterator stops when the shortest iterable is exhausted. For cases where the function inputs are already arranged into argument tuples, see
itertools.starmap()
.
map()
这个函数的第一个参数,就是用来接收函数的。随后的参数,是 iterable
—— 就是可被迭代的对象,比如,各种容器,例如:列表、元组、字典什么的。
def double_it(n):
return n * 2
a_list = [1, 2, 3, 4, 5, 6]
b_list = list(map(double_it, a_list))
print(b_list)
c_list = list(map(lambda x: x * 2, a_list))
print(c_list)
[2, 4, 6, 8, 10, 12]
[2, 4, 6, 8, 10, 12]
显然用 lambda
更为简洁。另外,类似完成 double_it(n)
这种简单功能的函数,常常有 “用过即弃” 的必要。
phonebook = [
{
'name': 'john',
'phone': 9876
},
{
'name': 'mike',
'phone': 5603
},
{
'name': 'stan',
'phone': 6898
},
{
'name': 'eric',
'phone': 7898
}
]
print(phonebook)
print(list(map(lambda x: x['name'], phonebook)))
print(list(map(lambda x: x['phone'], phonebook)))
[{'name': 'john', 'phone': 9876},
{'name': 'mike', 'phone': 5603},
{'name': 'stan', 'phone': 6898},
{'name': 'eric', 'phone': 7898}]
['john', 'mike', 'stan', 'eric']
[9876, 5603, 6898, 7898]
可以给 map() 传递若干个可被迭代对象:
a_list = [1, 3, 5]
b_list = [2, 4, 6]
print(list(map(lambda x, y: x * y, a_list, b_list)))
[2, 12, 30]
以上的例子都弄明白了,再去看 The Python Tutorial 中的例子,就不会有任何疑惑了:
pairs = [(1, 'one'), (2, 'two'), (3, 'three'), (4, 'four')]
pairs.sort(key=lambda p: p[1])
print(pairs)
[(4, 'four'), (1, 'one'), (3, 'three'), (2, 'two')]