Python3入门(六)——函数式编程

更多实例:参考:https://blog.csdn.net/u013398034/article/details/78701714

一、高阶函数

  1.可以通过变量指向函数,达到类似别名的效果:

>>> f = abs
>>> f(-10)
10

  2.函数的参数可以是函数,也就是函数可以作为一个入参

def add(x, y, f):
    return f(x) + f(y)

  以下介绍几个高阶函数

  map、reduce

    这两个函数就不赘述了。和scala的功能类似,不过用法不一样,它接收两个参数,第一个是函数f,第二个是Iterable。并将计算结果作为新的Iterator返回,惰性的Iterator通过list便可以计算出结果了:

def f(x):
    return x + 1


r = map(f, [1, 2, 3, 4, 5])
print(list(r))

    reduce入参类似map,但是它的f是接收两个参数,并且需要导包。并将结果作为下一次的输入,表示起来就是:

reduce(f, [x1, x2, x3, x4]) = f(f(f(x1, x2), x3), x4)
from functools import reduce


def f(x, y):
    return x + y


r = reduce(f, [1, 2, 3, 4, 5])
# 注意这里不需要list包装了,因为返回的不是Iterator
print(r)

    filter

  和scala的filter也是类似了,返回Ture的留下,其它过滤掉。其他的入参和返回值与map相同

def f(x):
    return x % 2 == 0


r = filter(f, [1, 2, 3, 4, 5])
print(list(r))

    sorted

    sorted可以直接对list进行比较

>>> sorted([-1, 3, 2, 6])
[-1, 2, 3, 6]

    也可以传入一个key的函数进行处理:

    key指定的函数将作用于list的每一个元素上,并根据key函数返回的结果进行排序。

sorted([36, 5, -12, 9, -21], key=abs)

    如果是对字符串比较,默认是比较ASCII码,也就是Z<a;要忽略大小写,可以通过传入的key函数对每个元素进行操作:

sorted(['img', 'ang', 'Zbb'], key=str.lower)
['ang', 'img', 'Zbb']

    反向排序,传入reverse参数

sorted(['img', 'ang', 'Zbb'], key=str.lower, reverse=True)
['Zbb', 'img', 'ang']

    一个简单的排序小示例如下:

# -*- coding: utf-8 -*-
L = [('Bob', 75), ('Adam', 92), ('Bart', 66), ('Lisa', 88)]


def by_name(t):
    return t[0].lower()


def by_score(t):
    return t[1]


L2 = sorted(L, key=by_name)
L3 = sorted(L, key=by_score, reverse=True)
print(L2)
print(L3)

二、返回函数

   高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回。

  以惰性求和为例,传统的求和是返回结算的结果,而高阶函数可以返回一个计算的函数,当调用这个返回的函数时才真正计算结果:

# -*- coding: utf-8 -*-
def my_sum(*args):
    def sum_in():
        s = 0
        for i in args:
            s += i
        return s

    return sum_in

  直接调用返回计算函数:

f = my_sum(1,2,3,4)
f
<function my_sum.<locals>.sum_in at 0x000002C607E15048>

  f才是真正的计算函数:

f()
10

  // 这里返回的内部函数sum_in引用了外部函数my_sum的参数和局部变量,这也叫Python中的闭包!

三、匿名函数

  也就是我们熟悉的拉姆达表达式了:lambda。Python中的用法如下:(需要显式使用lambda关键字)

list(map(lambda x:x*x, [1,2,3]))
[1, 4, 9]

  其他的lambda表达式不再赘述,和Java、scala一致

四、装饰器

  之前提到的,可以通过变量给函数起别名:

def fun1():
    print("100分,100分!")
    
f1 = fun1

  真实名字可以通过.__name__得出:

f1.__name__
'fun1'

  装饰器在其他语言中也比较常见了,这里介绍Python的装饰器。它本质上是一个返回函数的高阶函数:

    使用的方式是通过注解形式进行装饰,而装饰者Log这里接收一个函数,并返回一个函数

# -*- coding: utf-8 -*-
def log(fun):
    def wrapper(*args, **kv):
        print("被装饰的函数是:%s" % fun.__name__)
        return fun(*args, **kv)
    return wrapper


@log
def func():
    print("现在是2018-04-24")


func()

  至于装饰者本身需要参数的情况,参考廖老师的博客

五、偏函数

  偏函数有点儿类似原来的默认参数,就是把函数的某些参数给固定住。这样就能快速调用:

    偏函数的功能由functools提供,需要提前导包:

# -*- coding: utf-8 -*-
import functools

int2 = functools.partial(int, base=2)

int2('11001001')

  以上的偏函数就把int函数的base参数给固定住了,这样int2在进行字符串转换时就会默认转换为2进制了,而不用每次都通过int(str, base=2)来指定了!

 

posted @ 2018-04-23 21:24  ---江北  阅读(422)  评论(0编辑  收藏  举报
TOP