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中的XML 工具
Apr 13 Python
Python入门_学会创建并调用函数的方法
May 16 Python
Python实现针对给定字符串寻找最长非重复子串的方法
Apr 21 Python
python+Splinter实现12306抢票功能
Sep 25 Python
使用django-guardian实现django-admin的行级权限控制的方法
Oct 30 Python
pthon贪吃蛇游戏详细代码
Jan 27 Python
Django重置migrations文件的方法步骤
May 01 Python
Python字典对象实现原理详解
Jul 01 Python
TensorFlow实现指数衰减学习率的方法
Feb 05 Python
解决python图像处理图像赋值后变为白色的问题
Jun 04 Python
keras打印loss对权重的导数方式
Jun 10 Python
python中Matplotlib绘制直线的实例代码
Jul 04 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 文件上传模型,支持多文件上传
2009/08/13 PHP
php中static静态变量的使用方法详解
2010/06/04 PHP
ThinkPHP 连接Oracle数据库的详细教程[全]
2012/07/16 PHP
php设计模式之职责链模式实例分析【星际争霸游戏案例】
2020/03/27 PHP
动态表格Table类的实现
2009/08/26 Javascript
通过js为元素添加多项样式,浏览器全兼容写法
2014/08/30 Javascript
编程语言JavaScript简介
2014/10/16 Javascript
javascript实现画不相交的圆
2015/04/07 Javascript
angular.bind使用心得
2015/10/26 Javascript
js获取客户端操作系统类型的方法【测试可用】
2016/05/27 Javascript
jQuery表单验证简单示例
2016/10/17 Javascript
Vue2.0系列之过滤器的使用
2018/03/01 Javascript
vue-cli 如何打包上线的方法示例
2018/05/08 Javascript
Vue引入sass并配置全局变量的方法
2018/06/27 Javascript
微信小程序静默登录的实现代码
2020/01/08 Javascript
简介JavaScript错误处理机制
2020/08/04 Javascript
Python使用OpenCV进行标定
2018/05/08 Python
解决python测试opencv时imread导致的错误问题
2019/01/26 Python
Python3的unicode编码转换成中文的问题及解决方案
2019/12/10 Python
pytorch方法测试详解——归一化(BatchNorm2d)
2020/01/15 Python
全网首秀之Pycharm十大实用技巧(推荐)
2020/04/27 Python
pytorch 计算ConvTranspose1d输出特征大小方式
2020/06/23 Python
python3实现语音转文字(语音识别)和文字转语音(语音合成)
2020/10/14 Python
python opencv角点检测连线功能的实现代码
2020/11/24 Python
各大浏览器 CSS3 和 HTML5 兼容速查表 图文
2010/04/01 HTML / CSS
CSS3文本换行word-wrap解决英文文本超过固定宽度不换行
2013/10/10 HTML / CSS
HTML5拍照和摄像机功能实战详解
2019/01/24 HTML / CSS
加拿大折扣、优惠券和交易网站:WagJag
2018/02/07 全球购物
幼儿园运动会加油词
2014/02/14 职场文书
项目技术负责人岗位职责
2015/04/13 职场文书
志愿者服务活动总结报告
2015/05/06 职场文书
个人借条范本
2015/05/25 职场文书
个人欠条范本
2015/07/03 职场文书
2015年团委副书记工作总结
2015/07/23 职场文书
幼师自荐信范文(2016推荐篇)
2016/01/28 职场文书
Python图片检索之以图搜图
2021/05/31 Python