python 基础学习-12-函数三 装饰器和闭包

1. 高阶函数

• 接收函数作为参数,或者将函数作为返回值返回的函数就是高阶函数

#高阶函数
#特点1:接收一个或者多个函数作为函数
#特点2:将函数作为返回值返回

#接收函数作为参数
lst=[1,2,3,4,5,6,7,8,9,10]
#定义一个函数可以将指定的列表中所有的偶数,保存到一个新的列表中返回

def fn(lst):
    #参数 lst是要进行筛选的列表
    #创建一个新的列表
    new_lst=[]
    for n in lst:
        #判断n的奇偶
        if n%2==0:
            new_lst.append(n) #将数添加到列表中
    return new_lst
print(fn(lst))

[2, 4, 6, 8, 10]

判定是否为奇数:

#接收函数作为参数
lst=[1,2,3,4,5,6,7,8,9,10]
#定义一个函数可以将指定的列表中所有的偶数,保存到一个新的列表中返回

def fn(lst):
    #参数 lst是要进行筛选的列表
    #创建一个新的列表
    new_lst=[]
    for n in lst:
        #判断n的奇偶
        if n%2 !=0:
            new_lst.append(n) #将数添加到列表中
    return new_lst
print(fn(lst))
[1, 3, 5, 7, 9]

穿插多个条件在函数中,用函数来替代:

# #接收函数作为参数
lst=[1,2,3,4,5,6,7,8,9,10]
# #定义一个函数可以将指定的列表中所有的偶数,保存到一个新的列表中返回
#
def fn(lst):
    #参数 lst是要进行筛选的列表
    #定义一个函数,用来检测一个偶数
    def fn2(i):
        if i%2==0:
            return True
    #定义一个函数用来检测是否大于5
    def fn3(i):
        if i >5:
            return True
    #创建一个新的列表
    new_lst=[]
    for n in lst:
        #判断n的奇偶
        if not fn3(n):
            new_lst.append(n) #将数添加到列表中
    return new_lst
print(fn(lst))
[1, 2, 3, 4, 5]

2. 闭包

• 将函数作为返回值也是高阶函数我们也称为闭包

#将函数作为返回值返回,也成为闭包
def fn():
    #在函数内部定义一个函数
    def fn2():
        print("我是fn2")
    #将内部函数作为返回值返回
    return fn2
print(fn()) #结果是打印出的对象  <function fn.<locals>.fn2 at 0x100a3d620>
#将函数作为返回值返回,也成为闭包
def fn():
    #在函数内部定义一个函数
    def fn2():
        print("我是fn2")
    #将内部函数作为返回值返回
    return fn2
#r是一个函数,是调用fn()后返回的函数,这个函数是在fn()内部定义,并不是全局函数
r=fn()
r()
我是fn2

# #将函数作为返回值返回,也成为闭包                                                                             
# def fn():                                                                                     
#     #在函数内部定义一个函数                                                                              
#     def fn2():                                                                                
#         print("我是fn2")                                                                        
#     #将内部函数作为返回值返回                                                                             
#     return fn2                                                                                
# #r是一个函数,是调用fn()后返回的函数,这个函数是在fn()内部定义,并不是全局函数                                                  
# r=fn()                                                                                        
# r()                                                                                           
#通过闭包可以创建一些只有当前函数能访问的变量                                                                         
#可以将一些私有数据藏到闭包中                                                                                 
                                                                                                
def make_average():                                                                             
    nums=[] #此时nums是个全局变量                                                                       
    #定义一个函数 计算平均值                                                                               
                                                                                                
    def average(n):                                                                             
        #将n添加到列表中                                                                               
        nums.append(n)                                                                          
        #求平均值                                                                                   
        return sum(nums)/len(nums)                                                              
    return average                                                                              
a=make_average()                                                                                
print(a(10))                                                                                    
                                                                                                
nums=[]                                                                                         
print(a(20))                                                                                    
                                                                                                
                                                                                                 
10.0
15.0

• 闭包的好处
• 通过闭包可以创建一些只有当前函数能访问的变量
• 可以将一些私有数据藏到闭包中
• 行成闭包的条件:

  • 函数嵌套
  • 将内部函数作为返回值返回
  • 内部函数必须要使用到外部函数的变量

3. 装饰器的引入

• 我们可以直接通过修改函数中的代码来完成需求,但是会产生以下一些问题

#求任意两个数的和
def add(a,b):
    # print("开始计算")
    # print("计算结束")
    return a+b
#求任意两个数的积
def mul(a,b):
    return a*b

r=add(1,20)
print(r)
#我们可以直接通过修改函数中的代码来完成这个需求,但是会产生一些问题
#1.如果需要修改的函数比较多,修改起来比较麻烦
#2.不方便后期维护
#3.这样做会违反开闭原则(OCP)程序的设计 要求开发对程序的扩展,要关闭对程序的修改

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

def fn():
    print("我是fn函数")

#定义一个函数,对fn进行扩展
def fn2():
    print("函数开始执行")
    fn()
    print("函数执行结束")
fn2()
21
函数开始执行
我是fn函数
函数执行结束

• 如果修改的函数多,修改起来会比较麻烦
• 不方便后期的维护
• 这样做会违反开闭原则(ocp)
• 程序的设计,要求开发对程序的扩展,要关闭对程序的修改

4. 装饰器的使用

• 通过装饰器,可以在不修改原来函数的情况下来对函数进行扩展

def fn():
    print("我是fn函数")
def add (a,b):
    return a+b


def start_end(old):
    #用来对其他的函数进行扩展 扩展函数执行的时候 打印开始执行 执行后打印执行
    #参数old要扩展的函数对象
    #创建一个函数
    def new_function(a,b):
        print("开始执行")
        #要调用被扩展的函数
        result=old(a,b)
        print("执行结束")
        return result
    #返回新函数
    return new_function
f=start_end(add)

r= f(1,2)
print(r)
开始执行
执行结束
3
def fn():
    print("我是fn函数")
def add (a,b):
    return a+b


def start_end(old):
    #用来对其他的函数进行扩展 扩展函数执行的时候 打印开始执行 执行后打印执行
    #参数old要扩展的函数对象
    #创建一个函数
    def new_function(*args,**kwarges):
        #*args接收所有的位置参数, **kwarges接收所有的关键字参数
        #装包,把参数包成一个一个元组或一个一个字典
        print("开始执行")
        #要调用被扩展的函数
        result=old(*args,**kwarges)
        # 把位置参数拆包,放在一个元组进去,或者把一个关键字拆包
        print("执行结束")
        return result
    #返回新函数
    return new_function
f=start_end(fn)

r= f()
# print(r)
开始执行
我是fn函数
执行结束

• 在开发中,我们都是通过装饰器来扩展函数的功能的

def fn():
    print("我是fn函数")
def add (a,b):
    return a+b


def start_end(old):
    #用来对其他的函数进行扩展 扩展函数执行的时候 打印开始执行 执行后打印执行
    #参数old要扩展的函数对象
    #创建一个函数
    def new_function(*args,**kwarges):
        #*args接收所有的位置参数, **kwarges接收所有的关键字参数
        #装包,把参数包成一个一个元组或一个一个字典
        print("开始执行")
        #要调用被扩展的函数
        result=old(*args,**kwarges)
        # 把位置参数拆包,放在一个元组进去,或者把一个关键字拆包
        print("执行结束")
        return result
    #返回新函数
    return new_function
f=start_end(add)

r= f(555,666)
print(r)

#若以后对某一函数进行扩展,直接调用start_end, 最终都会返回一个新的函数,都是对原有的函数进行
#扩展,此时start_end就是一个装饰器

#向我们今天讲的这个start_end(old)类似于这样的函数我们就称之为装饰器
#通过装饰器,可以在不修改原来的函数的情况下来对函数进行扩展
#在开发中,我们都是通过装饰器来扩展函数的功能的

@start_end
def speak():
    print("大家加油")

speak()
开始执行
执行结束
1221
开始执行
大家加油
执行结束
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值