支持函数式编程的包

支持函数式编程的包

Python的目标不是变成函数式语言,但是得益于operator和functools等包的支持,函数式编程风格也可以信手拈来

operator模块

在函数式编程中,经常需要把算术运算符当作函数使用。例如,不适使用递归计算阶乘。求和可以使用sum函数,但是求积没有这样的函数。可以使用reduce函数,但是需要一个函数计算序列中两个元素之积

# 使用reduce函数和一个匿名函数计算阶乘
from functools import reduce


def fact(n):
    return reduce(lambda a, b: a * b, range(1, n + 1))

operator模块为多个算术运算符提供了对应的函数,从而避免编写lambda a, b:a*b这种平凡的匿名函数。使用算术运算符

from functools import reduce
from operator import mul


def fact(n):
    return reduce(mul, range(1, n + 1))

operator模块中还有一类函数,能替代从序列中取出元素或读取对象属性的lambda表达式:因此,itemgetter和attrgetter其实会自行构建函数

metro_data = [('Tokyo', 'JP', 36.933, (35.689722, 139.69167)),
              ('Delhi NCR', 'IN', 21.935, (28.613889, 77.208889))]
from operator import itemgetter

for city in sorted(metro_data, key=itemgetter(1)):
    print(city)
# itemgetter(1)的作用与lambda fields:fields[1]一样,创建一个接受集合的函数,返回索引为1上的元素
('Delhi NCR', 'IN', 21.935, (28.613889, 77.208889))
('Tokyo', 'JP', 36.933, (35.689722, 139.69167))
cc_name = itemgetter(1, 0)
for city in metro_data:
    print(cc_name(city))
# 如果把多个参数传给itemgetter,它构建的函数会返回提取的值构成的元组
('JP', 'Tokyo')
('IN', 'Delhi NCR')

itemgetter使用[]运算符,因此它不仅支持序列,还支持映射和任何实现__getitem__方法的类

attrgetter与itemgetter作用类似,它会创建函数根据名称提取对象的属性。如果把多个属性名传递给attrgetter,它也会返回提取的值构成的元组。此外,如果参数名中包含点号,attrgetter会深入嵌套对象,获取指定的属性。

# 定义一个namedtuple,名为metro_data,用attrgetter处理它
from collections import namedtuple

LatLong = namedtuple('LatLong', 'lat long')  # 使用namedtuple定义LatLong
Metropolis = namedtuple('Metropolis', 'name cc pop coord')  # 定义Metropolis
metro_areas = [Metropolis(name, cc, pop, LatLong(lat, long)) for name, cc, pop, (lat, long) in
               metro_data]  # 使用Metropolis实例构建metro_areas列表。使用嵌套的元组拆包提取(lat, long),然后使用它们构建LatLong,作为Metropolis的coord属性
metro_areas[0]
Metropolis(name='Tokyo', cc='JP', pop=36.933, coord=LatLong(lat=35.689722, long=139.69167))
metro_areas[0].coord.lat  # 深入metro_areas[0]获取它的纬度
35.689722
from operator import attrgetter

name_lat = attrgetter('name', 'coord.lat')  # 定义一个attrgetter,获取name属性和嵌套的coord.lat属性
for city in sorted(metro_areas, key=attrgetter('coord.lat')):  # 再次使用attrgetter,按照纬度排序城市列表
    print(name_lat(city))  # 使用定义的attrgetter,只显示城市名和纬度
('Delhi NCR', 28.613889)
('Tokyo', 35.689722)
# 下面是operator模块中定义的部分函数(省略了以_开头的函数名称,因为它们基本上是实现细节)
import operator

[name for name in dir(operator) if not name.startswith('_')]
['abs',
 'add',
 'and_',
 'attrgetter',
 'concat',
 'contains',
 'countOf',
 'delitem',
 'eq',
 'floordiv',
 'ge',
 'getitem',
 'gt',
 'iadd',
 'iand',
 'iconcat',
 'ifloordiv',
 'ilshift',
 'imatmul',
 'imod',
 'imul',
 'index',
 'indexOf',
 'inv',
 'invert',
 'ior',
 'ipow',
 'irshift',
 'is_',
 'is_not',
 'isub',
 'itemgetter',
 'itruediv',
 'ixor',
 'le',
 'length_hint',
 'lshift',
 'lt',
 'matmul',
 'methodcaller',
 'mod',
 'mul',
 'ne',
 'neg',
 'not_',
 'or_',
 'pos',
 'pow',
 'rshift',
 'setitem',
 'sub',
 'truediv',
 'truth',
 'xor']

这些函数中的大部分作用不言而喻。以i开头,后面是另一个运算符的那些名称(如iadd、iand等),对应的是增量赋值运算符(如+=、&=等)。如果第一个参数是可变的,那么这些运算符会就地修改它;否则,作用与不带i的函数一样,直接返回运算结果

在operator模块中,methodcaller的作用与attrgetter和itemgetter类似,它会自行创建函数。methodcaller创建的函数会在对象上调用参数指定的方法

from operator import methodcaller

s = 'The time has come'
upcase = methodcaller('upper')
upcase(s)
'THE TIME HAS COME'
hiphenate = methodcaller('replace', ' ', '-')  # methodcaller可以冻结某些参数
hiphenate(s)
'The-time-has-come'

使用functools.partial冻结参数

functools模块提供了一系列高阶函数,其中最为人熟知的是reduce函数

functools.partial这个高阶函数用于部分应用一个函数。部分应用是指,基于一个函数创建一个新的可调用对象,把原函数的某些参数固定。使用这个函数可以把接受一个或多个参数的函数改编成需要回调的API,这样参数更少

# 使用partial把两个参数函数改编成需要单参数的可调用对象
from operator import mul
from functools import partial

triple = partial(mul, 3) # 使用mul创建triple函数,把第一个定位参数定为3
triple(7) # 测试triple函数
21
list(map(triple,range(1,10))) # 在map中使用triple函数
[3, 6, 9, 12, 15, 18, 21, 24, 27]
# 使用partial构建一个便利的Unicode规范化函数
import unicodedata ,functools
nfc=functools.partial(unicodedata.normalize,'NFC')
# partial的第一个参数是可调用对象,后面跟着任意个要绑定的定位参数和关键字
上一篇:安装ROS操作系统


下一篇:MahApps.Metro 报找不到资源MahApps.Metro;component/Styles/Accents/BaseLight.xaml