在python中,装饰器、生成器和迭代器是特别重要的高级函数   https://www.cnblogs.com/yuanchenqi/articles/5830025.html

装饰器

  1.如果说装饰器是取经路上的大妖怪,想要干掉它就必须拿到三样法宝

法宝一:作用域(LEGB)

法宝二:高阶函数:(把函数名看作变量,可以作为传输参数,也可以作为返回值)

(在python的世界里,函数和我们之前的[1,2,3],'abc',8等一样都是对象,

而且函数是最高级的对象(对象是类的实例化,可以调用相应的方法,函数是包含变量对象的对象,牛逼!)。)

法宝三:闭包 (LEGB规则的特例,闭包实现了enclosing变量在enclosing范围之外的使用)

闭包(closure)是函数式编程的重要的语法结构。

定义:如果在一个内部函数里(local),对在外部作用域(但不是在全局作用域)的变量(enclosing)进行引用,

   那么内部函数就被认为是闭包(closure).

   闭包=函数块+定义函数时的环境,inner就是函数块,x就是环境,当然这个环境可以有很多,不止一个简单的x

def f_outer():
x=10 #enclosing变量(外部环境)
def f_inner(): #内部函数(local)
print (x)
return f_inner #enclosing变量 f_outer()() 输出结果:10

闭包(closure)

2.装饰器(从实现功能出发理解)

装饰器本质上是一个函数,该函数用来处理其他函数,它可以让其他函数在不需要修改代码的前提下增加额外的功能,装饰器的返回值也是一个函数对象。它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等应用场景。装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。

概括的讲,装饰器的作用就是在不修改源代码的情况下为已经存在的对象添加额外的功能

1.重新定义一个函数,为原函数添加功能

2.需要外加一个装饰器(函数)用来传递原函数,然后返回step1中定义的函数

3.如果还需要传其它参数,可以再外加一个装饰器传递参数

简单装饰器的实现

if  foo()==show_time(foo) :问题解决!

所以,我们需要show_time(foo)返回一个函数对象,而这个函数对象内则是核心业务函数:执行func()与装饰函数时间计算,修改如下:

函数show_time就是装饰器,它把真正的业务方法func包裹在函数里面,看起来像foo被上下时间函数装饰了。在这个例子中,函数进入和退出时 ,被称为一个横切面(Aspect),这种编程方式被称为面向切面的编程(Aspect-Oriented Programming)。

 import time
def foo():
print('Function1')
time.sleep(5) def bar():
print('Function2')
time.sleep(5) def show_time(func):
def inner():
start=time.time()
func()
end=time.time()
print(end-start)
return inner
foo=show_time(foo)
bar=show_time(bar) foo()
bar()

闭包实现

@符号是装饰器的语法糖,在定义函数的时候使用,避免再一次赋值操作

 import time
def show_time(func):
def inner():
start=time.time()
func()
end=time.time()
print(end-start)
return inner @show_time
def foo():
print('Function1')
time.sleep(5) @show_time
def bar():
print('Function2')
time.sleep(5) # foo=show_time(foo)
# bar=show_time(bar) foo()
bar()
输出结果:
Function1
5.000344753265381
Function2
5.00098443031311

@

如上所示,这样我们就可以省去bar = show_time(bar)这一句了,直接调用bar()即可得到想要的结果。如果我们有其他的类似函数,我们可以继续调用装饰器来修饰函数,而不用重复修改函数或者增加新的封装。这样,我们就提高了程序的可重复利用性,并增加了程序的可读性。

这里需要注意的问题:  foo=show_time(foo)其实是把wrapper引用的对象引用给了foo,而wrapper里的变量func之所以可以用,就是因为wrapper是一个闭包函数。

(从宏观的角度理解)

带参数的被装饰函数

 import time
def show_time(func):
def inner(x,y):
start=time.time()
func(x,y)
end=time.time()
print(end-start)
return inner
@show_time
def add(a,b):
print(a+b)
time.sleep(5) add(2,3)
输出结果:
5

定长参数

 import time
def show_time(func):
def inner(*args,**kwargs):
start=time.time()
func(*args)
end=time.time()
print(end-start)
return inner
@show_time
def add(*args,**kwargs):
sum=0
for i in args:
sum+=i
print(sum)
time.sleep(5) add(2,3,4,5,6) 输出结果:
20

不定长参数

 带参数的装饰器

装饰器还有更大的灵活性,例如带参数的装饰器:

在上面的装饰器调用中,比如@show_time,该装饰器唯一的参数就是执行业务的函数。

装饰器的语法允许我们在调用时,提供其它参数,比如@decorator(a)。

 import time
def time_logger(flag=0):
def show_time(func):
def inner(*args,**kwargs):
start=time.time()
func(*args,**kwargs)
end=time.time()
print(end-start)
if flag:
print('打印日志')
return inner
return show_time
@time_logger(3)
def add(*args,**kwargs):
sum=0
for i in args:
sum+=i
print(sum)
time.sleep(5) add(2,3,4,5,6) 输出结果:
20
5.000020742416382
打印日志

带参数的装饰器

(说明可以通过装饰器给闭包传输参数)

3.装饰器的应用举例之登录

Day11 Python基础之装饰器(高级函数)(九)的更多相关文章

  1. 十. Python基础(10)--装饰器

    十. Python基础(10)--装饰器 1 ● 装饰器 A decorator is a function that take a function as an argument and retur ...

  2. 1.16 Python基础知识 - 装饰器初识

    Python中的装饰器就是函数,作用就是包装其他函数,为他们起到修饰作用.在不修改源代码的情况下,为这些函数额外添加一些功能,像日志记录,性能测试等.一个函数可以使用多个装饰器,产生的结果与装饰器的位 ...

  3. python基础之 装饰器,内置函数

    1.闭包回顾 在学习装饰器之前,可以先复习一下什么是闭包? 在嵌套函数内部的函数可以使用外部变量(非全局变量)叫做闭包! def wrapper(): money =10 def inner(num) ...

  4. [python基础]关于装饰器

    在面试的时候,被问到装饰器,在用的最多的时候就@classmethod ,@staticmethod,开口胡乱回答想这和C#的static public 关键字是不是一样的,等面试回来一看,哇,原来是 ...

  5. 【Python成长之路】python 基础篇 -- 装饰器【华为云分享】

    [写在前面] 有时候看到大神们的代码,偶尔会用到@来装饰函数.当时查了资料,大致了解装饰器一般用于在不改变原函数的基础上 ,对原函数功能进行修改/增强.使用场景是:日志级别设置.权限校验.性能测试等. ...

  6. python基础(8)-装饰器函数&进阶

    从小例子进入装饰器 统计一个函数执行耗时 原始版本 import time # time模块有提供时间相关函数 def do_something(): print("do_something ...

  7. python基础之装饰器(实例)

    1.必备 #### 第一波 #### def foo(): print 'foo' foo #表示是函数 foo() #表示执行foo函数 #### 第二波 #### def foo(): print ...

  8. 【Python基础】装饰器的解释和用法

    装饰器的用法比较简单,但是理解装饰器的原理还是比较复杂的,考虑到接下来的爬虫框架中很多用到装饰器的地方,我们先来讲解一下. 函数 我们定义了一个函数,没有什么具体操作,只是返回一个固定值 请注意一下缩 ...

  9. 学习PYTHON之路, DAY 5 - PYTHON 基础 5 (装饰器,字符格式化,递归,迭代器,生成器)

    ---恢复内容开始--- 一 装饰器 1 单层装饰器 def outer(func): def inner(): print('long') func() print('after') return ...

随机推荐

  1. 老K漫谈区块链的共识(1)——免信任的共识机制

    老k,柏链道捷CTO.清华阿尔山区块链研究中心高级工程师,超过17年的系统软件开发经验,在操作系统.编译器.虚拟机和符号执行方面都有实战经验.主持开发多个开眼项目,目前主要从事区块链底层系统开发工作. ...

  2. Mybatis 学习---${ }与#{ }获取输入参数的区别、Foreach的用法

    一.Mybatis中用#{}和${}获取输入参数的区别 1.“#{}“和“${}”都可以从接口输入中的map对象或者pojo对象中获取输入的参数值.例如 <mapper namespace=&q ...

  3. 函数指针的返回值是指针数组,数组里放的是int;函数指针的返回值是指针数组,数组里放的是int指针

    函数指针的返回值是指针数组,数组里放的是int 函数指针的返回值是指针数组,数组里放的是int指针 #include <stdio.h> #include <stdlib.h> ...

  4. windows7家庭版,专业版,旗舰版,企业版版本区别

    Windows 7包含6个版本,分别为Windows 7 Starter(初级版).Windows 7 Home Basic(家庭普通版).Windows 7 Home Premium(家庭高级版). ...

  5. LeetCode算法题-Happy Number(Java实现)

    这是悦乐书的第188次更新,第190篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第47题(顺位题号是202).编写算法以确定数字是否"幸福". 幸福 ...

  6. mysql统计一年12月的数据

    效果图: select end) as 一月份, end) as 二月份, end) as 三月份, end) as 四月份, end) as 五月份, end) as 六月份, end) as 七月 ...

  7. C#深度学习の委托深度解析

    一.我们在使用C#的过程中,不可避免的用到了委托. 委托的本质是什么呢? 从语法上看,委托是对方法的抽象封装,例如:public void print1(),public void print2(), ...

  8. 在django中如何通过已有的mysql表生成django的model

    第一步:先是在项目中建立一个app:python manage.py startapp app01 第二步:python manage.py inspectdb > app01/models.p ...

  9. zabbix图形乱码问题解决办法

    zabbix中的图形乱码的问题解决办法: 1.下载字体,例如:simkai.ttf楷体(注:在windows中的字体格式可能是TTC的,所以去网上下载一个ttf的字体) 2.上传到linux中(我使用 ...

  10. Python *args 和 **kwargs用法

    *args的参数形式把剩下的没有关键字的参数收起来形成一个tuple,*kwargs把有关键字的收起来做成一个字典 def this_fun(a,b,*args,**kwargs): print a ...