Python高级特性之闭包与装饰器实例详解


Posted in Python onNovember 19, 2019

本文实例讲述了Python高级特性之闭包与装饰器。分享给大家供大家参考,具体如下:

闭包

1.函数参数:

(1)函数名存放的是函数的地址
(2)函数名()存放的是函数内的代码
(3)函数名只是函数代码空间的引用,当函数名赋值给一个对象的时候,就是引用传递

def func01():
  print("func01 is show")
test = func01
print(func01)
print(test)
test()

结果:

Python高级特性之闭包与装饰器实例详解

2.闭包:

(1)内层函数可以访问外层函数变量

(2)闭包就是一个嵌套定义的函数,在外层运行时才开始内层函数的定义,然后将内部函数的引用传递函数外的对象(闭包外层函数的返回值为内层函数名)

(3)内部函数和使用的外部函数提供的变量构成的整体称为闭包

def func_out(rate):
  def func_in(money):
    print(rate * money)
  return func_in
usa_money = func_out(0.7)
usa_money(100)
usa_money(200)

执行结果:

Python高级特性之闭包与装饰器实例详解

装饰器

装饰器就是在不改变函数的原有代码的前提下 给函数添加新的功能,装饰器一般是一个闭包。

1.装饰器:

# 在不改变函数的原有代码的前提下 给函数添加新的功能
def func_out(func):
  def func_in():
    print("验证")
    func()
  return func_in
@func_out
def login():
  print("登录")
# 有装饰器 装饰器下面正好是一个函数
# login = func_out(login)  重点
login()

如果@func_out装饰器下刚好是一个login函数,会执行:

login = func_out(login)

所以会直接执行func_out内的代码,func为原login,即指向原login函数的地址空间并且返回内层函数名,即:

login = func_out(login) = func_in

所以login()的执行结果为func_in(),即:

print('验证')
func() # func指原login函数的地址

就实现了不改变原函数的情况下给函数添加新功能

执行结果:

Python高级特性之闭包与装饰器实例详解

2.装饰有返回指函数:

def func_out(func):
  def func_in():
    # ret = func()
    #  def login():
    #    return 100
    return func()
  return func_in
@func_out
def login():
  return 100
# login() ==> func_in()
# func ==> 原始的login
f = login()
print(f)

3.装饰有参数函数:

def func_out(func):
  def func_in(a):
    func(a)
  return func_in
@func_out
def login(a):
  print(a)
# login() ==> func_in()
# func ==> 原始的login
login(10)

4.装饰器通用版:

def func_out(func):
  def func_in(*args,**kwargs):
    return func(*args,**kwargs)
  return func_in
@func_out
def login(*args,**kwargs):
  print(args)
  print(kwargs)
# login() ==> func_in()
# func ==> 原始的login
login(10,20,age = "17",name="123")

5.类装饰器:

class Foo(object):
  def __init__(self, func):
    self.func = func
  def __call__(self):
    print("验证")
    self.func()
@Foo
def login():
  print("登录")
# login = Foo(login)
login()

6.多装饰器:

def func_out01(func01):
  print("func_out01 is show")
  def func_in01():
    print("func_in01 is show")
    func01()
  return func_in01
def func_out02(func02):
  print("func_out02 is show")
  def func_in02():
    print("func_in02 is show")
    func02()
  return func_in02
@func_out02 # login = func_out02(login)
@func_out01 # login = func_out01(login)
def login():
  print("login is show")
login()

执行结果:

因为@闭包名下为函数时才会实现装饰器,所以func_out1会先装饰函数,func_out2会后装饰函数,所以外层函数先执行func_out1,后执行func_out2;因为func_out1先装饰函数,func_out2后装饰函数,所以func_out1装饰后,原函数为先输出func_in1内的语句,再输出原login,然后func_out2装饰后,执行顺序为先输出func_in2的语句,再输出装饰后的login函数,即:func_in2——func_in1——login。

Python高级特性之闭包与装饰器实例详解

7.给装饰器传递函数:

def route(参数):
  print(参数)
  def func_out(func):
    def func_in():
      func()
    return func_in
  return func_out
@route(参数)
def index():
  return "index is show"

希望本文所述对大家Python程序设计有所帮助。

Python 相关文章推荐
python在linux中输出带颜色的文字的方法
Jun 19 Python
Python捕捉和模拟鼠标事件的方法
Jun 03 Python
Python设计模式之外观模式实例详解
Jan 17 Python
Python3.5基础之NumPy模块的使用图文与实例详解
Apr 24 Python
Python中typing模块与类型注解的使用方法
Aug 05 Python
Python数据可视化 pyecharts实现各种统计图表过程详解
Aug 15 Python
详解tensorflow2.x版本无法调用gpu的一种解决方法
May 25 Python
使用已经得到的keras模型识别自己手写的数字方式
Jun 29 Python
Matplotlib.pyplot 三维绘图的实现示例
Jul 28 Python
Django前后端分离csrf token获取方式
Dec 25 Python
虚拟环境及venv和virtualenv的区别说明
Feb 05 Python
如何使用PyCharm及常用配置详解
Jun 03 Python
Python高级编程之继承问题详解(super与mro)
Nov 19 #Python
Python3 Tkinkter + SQLite实现登录和注册界面
Nov 19 #Python
Python csv文件的读写操作实例详解
Nov 19 #Python
Python高级property属性用法实例分析
Nov 19 #Python
wxPython之wx.DC绘制形状
Nov 19 #Python
python matplotlib拟合直线的实现
Nov 19 #Python
Python线程指南分享
Nov 19 #Python
You might like
PHP连接Access数据库的方法小结
2013/06/20 PHP
php缓冲输出实例分析
2015/01/05 PHP
Laravel5中contracts详解
2015/03/02 PHP
PHPExcel简单读取excel文件示例
2016/05/26 PHP
PHP删除数组中指定值的元素常用方法实例分析【4种方法】
2018/08/21 PHP
YII2框架中使用RBAC对模块,控制器,方法的权限控制及规则的使用示例
2020/03/18 PHP
javascript下判断一个元素是否存在的代码
2010/03/05 Javascript
jquery中EasyUI实现同步树
2015/03/01 Javascript
详解Angularjs中的依赖注入
2016/03/11 Javascript
js获取时间函数及扩展函数的方法
2016/10/30 Javascript
微信小程序 loading 详解及实例代码
2016/11/09 Javascript
node学习记录之搭建web服务器教程
2017/02/16 Javascript
原生JS实现的碰撞检测功能示例
2018/05/18 Javascript
vue实现抽屉弹窗效果
2020/11/15 Javascript
Python with的用法
2014/08/22 Python
python中numpy.zeros(np.zeros)的使用方法
2017/11/07 Python
python实现音乐下载的统计
2018/06/20 Python
Pycharm+Scrapy安装并且初始化项目的方法
2019/01/15 Python
用python建立两个Y轴的XY曲线图方法
2019/07/08 Python
Django+boostrap 美化admin后台的操作
2020/03/11 Python
Django admin 实现search_fields精确查询实例
2020/03/30 Python
如何表示python中的相对路径
2020/07/08 Python
理解Django 中Call Stack机制的小Demo
2020/09/01 Python
Hotter Shoes英国官网:英伦风格,舒适的鞋子
2017/12/28 全球购物
西班牙手机之家:Phone House
2018/10/18 全球购物
美国健康和保健平台:healtop
2020/07/02 全球购物
求职信需要的五点内容
2014/02/01 职场文书
读书小明星事迹材料
2014/05/03 职场文书
出差报告格式模板
2014/11/06 职场文书
2014年秘书工作总结
2014/11/25 职场文书
2014年小学美术工作总结
2014/12/20 职场文书
美容院员工规章制度
2015/08/05 职场文书
导游词之安徽九华山
2019/09/18 职场文书
2019年12月24日平安夜祝福语集锦
2019/12/24 职场文书
利用JuiceFS使MySQL 备份验证性能提升 10 倍
2022/03/17 MySQL
MySQL详解进行JDBC编程与增删改查方法
2022/06/16 MySQL