一、函数式编程

  • 在Python中,函数是一等对象
    - 一等对象一般都会具有如下特点:
    ① 对象是在运行时创建的
    ② 能赋值给变量或作为数据结构中的元素
    ③ 能作为参数传递
    ④ 能作为返回值返回
    就是将一个一个的功能通过函数来完成,python支持函数式编程,但python不是函数式编程语言。

二、什么是高阶函数

接收函数作为参数,或者将函数作为返回值的函数是高阶函数
当我们使用一个函数作为参数时,实际上是将指定的代码传递进了目标函数。

  • 高阶函数至少要符合以下两个特点中的一个
    ① 接收一个或多个函数作为参数
    ② 将函数作为返回值返回

为什么需要高阶函数呢?
例子:定义一个函数,可以将指定列表中的所有的偶数,或所有奇数,或大于5的数等,保存到一个新的列表中返回

用普通的函数实现

l = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
def fn(lst):
    # 创建一个新列表
    new_list = []
    # 对列表进行筛选
    for n in lst:
        # # 判断n的奇偶
        # if n % 2 == 0:
        #     new_list.append(n)
        # # 取奇数
        # if n % 2 != 0:
        #     new_list.append(n)
        # # 取大于5的数
        # if n > 5:
        #     new_list.append(n)
        if not fn2(n):
            new_list.append(n)

    # 返回新列表
    return new_list


print(fn(l))

或:

def fn(lst):
    def fn2(i):
        if i % 2 == 0:
            return True
        return False

    def fn3(i):
        if i > 5:
            return True
        return False

    # 创建一个新列表
    new_list = []
    # 对列表进行筛选
    for n in lst:
        # # 判断n的奇偶
        if fn2(n):
            new_list.append(n)
        # 大于5的数
        if fn3(n):
            new_list.append(n)

    # 返回新列表
    return new_list


print(fn(l))

使用高阶函数实现:

# 在函数内部在定义一个函数,用来检查一个任意数字是否是偶数
def fn2(i):
    if i % 2 == 0:
        return True
    return False


def fn3(i):
    if i > 5:
        return True
    return False


def fn(func, lst):
    # 创建一个新列表
    new_list = []
    # 对列表进行筛选
    for n in lst:
        if func(n):
            new_list.append(n)

    # 返回新列表
    return new_list

# 当传fn2函数作为参数时,实际上是将fn2的代码传递
print(fn(fn2, l)) # [2, 4, 6, 8, 10]
print(fn(fn3, l)) # [6, 7, 8, 9, 10]

三、filter()函数

filter()可以从序列中过滤出符合条件的元素,保存到一个新的序列中。
参数:
1.函数,根据该函数来过滤序列(可迭代的结构)
2.需要过滤的序列(可迭代的结构)
返回值:
过滤后的新序列(可迭代的结构)

l = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]


# 在函数内部在定义一个函数,用来检查一个任意数字是否是偶数
def fn2(i):
    if i % 2 == 0:
        return True
    return False


r = filter(fn2, l)
print(list(r))  #[2, 4, 6, 8, 10]

fn2是作为参数传递进filter()函数中
而fn2实际上只有一个作用,就是作为filter()的参数
filter()调用完毕以后,fn2就已经没用了。

四、匿名函数

匿名函数 lambda 函数表达式 (语法糖)
lambda函数表达式专门用来创建一些简单的函数,他是函数创建的又一种方式
语法:lambda 参数列表 : 返回值
匿名函数一般都是作为参数使用,其他地方一般不会使用.

def fn5(a, b):
    return a + b

print(fn5(123, 456))  # 579

# 用lambda来代替这个函数
lambda a, b: a + b
print((lambda a, b: a + b)(123,456))  # 579

# 将lambda表达式,赋值给一个变量
result  = lambda a, b: a + b
print(result(123,456)) #
def fn4(i):
    if i % 3 == 0:
        return True
    return False

# 第一次改造
def fn4(i):
    return i % 3 == 0


# 改造lambda
lambda i: i % 3 == 0

# r = filter(fn4(), l)
r = filter(lambda i: i % 3 == 0, l)
print(list(r))

五、map函数

map()函数可以对可跌倒对象中的所有元素做指定的操作,然后将其添加到一个新的对象中返回

l = [1,2,3,4,5,6,7,8,9,10]

r = map(lambda i : i+1 , l)
print(r)  # <map object at 0x7fec094ce0f0>
print(list(r)) # [2, 3, 4, 5, 6, 7, 8, 9, 10, 11]

六、sort函数

该方法用来对列表中的元素进行排序
sort()方法默认是直接比较列表中的元素的大小
在sort()可以接收一个关键字参数 , key
key需要一个函数作为参数,当设置了函数作为参数
每次都会以列表中的一个元素作为参数来调用函数,并且使用函数的返回值来比较元素的大小

l = ['bb','aaaa','c','ddddddddd','fff']
l.sort()
print(l) # ['aaaa', 'bb', 'c', 'ddddddddd', 'fff']


l = ['bb','aaaa','c','ddddddddd','fff']
l.sort(key=len)
print(l) # ['aaaa', 'bb', 'c', 'ddddddddd', 'fff']  # ['c', 'bb', 'fff', 'aaaa', 'ddddddddd']


l = [2,5,'1',3,'6','4']
l.sort(key=int) 
print(l)   # ['1', 2, 3, '4', 5, '6']

七、sorted()

这个函数和sort()的用法基本一致,但是sorted()可以对任意的序列进行排序
并且使用sorted()排序不会影响原来的对象,而是返回一个新对象

l = [2, 5, '1', 3, '6', '4']
print('排序前:', l)
print(sorted(l, key=int))
print('排序后:', l)   

控制台打印;
排序前: [2, 5, '1', 3, '6', '4']
['1', 2, 3, '4', 5, '6']
排序后: [2, 5, '1', 3, '6', '4']

八、闭包

将函数作为返回值返回,也是一种高阶函数
这种高阶函数我们也称为叫做闭包,通过闭包可以创建一些只有当前函数能访问的变量
可以将一些私有的数据藏到的闭包中

形成闭包的条件:
① 函数嵌套
② 将内部函数作为返回值返回
③ 内部函数必须要使用到外部函数的变量

# 外部函数
def make_averager():
    # 创建一个列表,用来保存数值
    nums = []

    # 创建内部函数,用来计算平均值
    def averager(n) :
        # 将n添加到列表中
        nums.append(n)
        # 求平均值
        return sum(nums)/len(nums)

    return averager

averager = make_averager()

print(averager(10))
print(averager(20))
print(averager(30))
print(averager(40))

九、装饰器

1、引入装饰器
举例: 希望函数可以在计算前,打印开始计算,计算结束后打印计算完毕

def add(a , b):
    '''
        求任意两个数的和
    '''
    print("开始计算...")
    r = a + b
    print('计算结束...')
    return r


def mul(a , b):
    '''
        求任意两个数的积
    '''
    print("开始计算...")
    r = a * b
    print('计算结束...')
    return r

弊端:如果多个函数需要打印计算开始和计算结束,那么就要给每一个函数都要添加两行语句。

我们希望在不修改原函数的情况下,来对函数进行扩展

def fn():
    print('我是fn函数....')

# 只需要根据现有的函数,来创建一个新的函数
def fn2():
    print('函数开始执行~~~')
    fn()
    print('函数执行结束~~~')

fn2()
控制台打印:
函数开始执行~~~
我是fn函数....
函数执行结束~~~


def new_add(a,b):
    print('计算开始~~~')
    r = add(a,b)
    print('计算结束~~~')
    return r

r = new_add(111,222)
print(r)

控制台打印结果:
计算开始~~~
计算结束~~~
333

将new_add变成一个通用的函数,不用对每一个需要扩展的函数,都要创建一个新的函数。

上边的方式,已经可以在不修改源代码的情况下对函数进行扩展了
但是,这种方式要求我们每扩展一个函数就要手动创建一个新的函数,实在是太麻烦了
为了解决这个问题,我们创建一个函数,让这个函数可以自动的帮助我们生产函数。

def add(a , b):
    '''
        求任意两个数的和
    '''
    r = a + b
    return r


def mul(a , b):
    '''
        求任意两个数的积
    '''
    r = a * b
    return r  



def begin_end(old):
    '''
        用来对其他函数进行扩展,使其他函数可以在执行前打印开始执行,执行后打印执行结束

        参数:
            old 要扩展的函数对象
    '''
    # 创建一个新函数  将参数装包成一个元组或者一个字典
    def new_function(*args , **kwargs):
        print('开始执行~~~~')
        # 调用被扩展的函数 ,将参数拆包成一个元组或一个字典
        result = old(*args , **kwargs)
        print('执行结束~~~~')
        # 返回函数的执行结果
        return result

    # 返回新函数        
    return new_function

f = begin_end(fn)
f2 = begin_end(add)
f3 = begin_end(mul)

# r = f()
# r = f2(123,456)
# r = f3(123,456)
# print(r)

像begin_end()这种函数我们就称它为装饰器
通过装饰器,可以在不修改原来函数的情况下来对函数进行扩展
在开发中,我们都是通过装饰器来扩展函数的功能的。(将旧函数作为参数传递给新的函数,新函数对旧函数进行升级)
在定义函数时,可以通过@装饰器,来使用指定的装饰器,来装饰当前的函数
可以同时为一个函数指定多个装饰器,这样函数将会安装从内向外的顺序被装饰

@begin_end
def say_hello():
    print('大家好~~~')

say_hello()

@fn3
@begin_end
def say_hello():
    print('大家好~~~')

say_hello()  
先执行begin_end装饰,在执行fn3装饰。