闭包函数

闭包概念

  • 闭:定义在函数内部的函数
  • 包:内部函数使用了外层函数名称空间中的名字
# 闭包函数
def outer():
x = 111
# 定义在函数内部的函数
def inner():
# 使用了外层函数名称空间中的名字
print(x)
return inner
x = 666
res = outer()
res() # 输出:111

实际应用

闭包函数是给函数体传参的另外一种方式

def outer(username):
def index():
print(username)
return index
res = outer('kevin') # 形参username与值kevin临时绑定
res() # 输出:kevin
res1 = outer('jason') # 形参username与值jason临时绑定
res1() # 输出:jason

这时候有人就很迷惑了,明明只要定义一个函数index()就可以了,为什么要用outer()包起来传参呢?别急,这是为接下来的内容做铺垫用的。

装饰器

简介

装饰器并不是一个新的知识点,而是由前面所有的函数知识点整合到一起的产物。

装饰器的本质:在不改变被装饰对象原有的“调用方式”和“内部代码”的情况下给被装饰对象添加新的功能。

这是不是听起来有些神奇?装饰器的原则就是对扩展开放,对修改封闭。

简单版本装饰器

了解完什么是装饰器之后,我们先学习使用简单版本的装饰器。

现在有一段代码:

import time  # 导入模块,不用管什么意思
def index():
time.sleep(1) # 让程序休眠1秒钟
print('这里是index函数') def home():
time.sleep(1) # 让程序休眠1秒钟
print('这里是home函数')

我如何能在不动上面2个函数的情况下,添加一个功能:输出函数运行的时长。

这时候有人就要说了:啊?这不是很简单,我只要添加一个函数

def get_time(function_name):
start_time = time.time() # 获取运行函数前的时间
function_name() # 调用函数
end_time = time.time() # 获取运行结束后的时间
print(end_time - start_time) # 输出时间差 get_time(index)
get_time(home)

这个方法确实是可以,但是运行函数都要调用get_time()。

所以别着急,装饰器的功能还没了解全之前,请先继续看下去。

实现:

import time  # 导入模块,不用管什么意思

# 装饰对象
def index():
time.sleep(1) # 让程序休眠1秒钟
print('这里是index函数') def home():
time.sleep(1) # 让程序休眠1秒钟
print('这里是home函数') # 装饰器
def outer(function_name):
def inner():
# 获取运行函数前的时间
start_time = time.time()
# 调用名为function_name函数
function_name()
# 获取函数运行结束后的时间
end_time = time.time()
# 输出时间差
print(end_time - start_time)
# 返回内部函数名
return inner
# 让index()变成inner()
index = outer(index)
index()
"""
输出:
这里是index函数
1.0101792812347412
""" # 狸猫换太子
home = outer(home)
home()
"""
输出:
这里是home函数
1.0047881603240967
"""

这个时候函数只需要调用它自己就能输出运行时间了,相当于给函数内部添加了一段代码一样。

进阶版本装饰器

这时候如果函数有参数该怎么做呢?并且参数还不知道几个?

很简单,只要加上可变长形参就行了。

实现:

import time

def index(a):  # 有参数
time.sleep(1)
print('这里是index函数', a) def home(): # 无参数
time.sleep(1)
print('这里是home函数') def outer(function_name):
def inner(*args, **kwargs): # 添加可变长形参
start_time = time.time()
function_name(*args, **kwargs) # 传参
end_time = time.time()
print(end_time - start_time)
return inner
index = outer(index)
index(555)
"""
输出:
这里是index函数 555
1.0047197341918945
""" home = outer(home)
home()
"""
输出:
这里是home函数
1.0003490447998047
"""

完整版本装饰器

这个时候问题又出现了,如果函数有返回值又该怎么办?

也很简单,在装饰器内部函数加一个return返回值就行了。

实现:

import time

def index(a):
time.sleep(1)
print('这里是index函数')
return a # 有返回值 def home():
time.sleep(1)
print('这里是home函数') def outer(function_name):
def inner(*args, **kwargs):
start_time = time.time()
# 将函数返回值赋值给res
res = function_name(*args, **kwargs)
end_time = time.time()
print(end_time - start_time)
"""返回函数的返回值"""
return res
return inner
index = outer(index)
print(index(555))
"""
输出:
这里是index函数
1.0073633193969727
555
""" home = outer(home)
home()
"""
输出:
这里是home函数
1.0051474571228027
"""

装饰器模板

还没搞明白装饰器怎么回事?小问题!这里有一套万能模板,只要知道怎么使用装饰器就行了。

'''编写装饰器其实有一套固定的代码 不需要做任何理解'''
def outer(func_name): # func_name用于接收被装饰的对象(函数)
def inner(*args, **kwargs):
print('执行被装饰函数之前 可以做的额外操作')
res = func_name(*args, **kwargs) # 执行真正的被装饰函数
print('执行被装饰函数之后 可以做的额外操作')
return res # 返回真正函数的返回值
return inner

装饰器语法糖

由于感觉在这块代码太繁琐了

index = outer(index)
index()

于是python给了一个方法,用@+装饰器名称放在装饰对象的上方。

实现:

def outer(func_name):
def inner(*args, **kwargs):
# 额外操作
res = func_name(*args, **kwargs)
# 额外操作
return res
return inner # 装饰器语法糖
@outer
def index():
print('这里是index函数') """可以直接使用index(),省去了一个步骤"""
index()

语法糖内部原理:

  1. 使用的时候最好紧跟在被装饰对象的上方
  2. 语法糖会自动将下面紧挨着的函数名传给@后面的函数调用

装饰器修复技术

在了解装饰器修复技术之前,我们先了解被装饰对象的内部属于谁了

def outer(func_name):
def inner(*args, **kwargs):
res = func_name(*args, **kwargs)
return res
return inner @outer
def index():
print('这里是index函数') print(index)
"""
输出:
<function outer.<locals>.inner at 0x0000025BA5B9F268>
"""

可以看出,这时的index已经是inner的人了,这就让人很不爽了,所以我们要把它恢复回来。

from functools import wraps  # 导入模块
def outer(func_name):
@wraps(func_name) # 修复
def inner(*args, **kwargs):
res = func_name(*args, **kwargs)
return res
return inner @outer
def index():
print('这里是index函数') print(index)
"""
输出:
<function index at 0x00000133924FF268>
"""

可以看到,index还是原来的index。

问题

编写一个用户认证装饰器:

  • 基本要求

    执行每个函数的时候必须先校验身份,eg: jason 123

  • 拔高练习

    执行被装饰的函数,只要有一次认证成功,那么后续的校验都通过

    提示:全局变量 记录当前用户是否认证

答案

点击查看代码(这里只展示简易版本)
# 定义全局变量,用于判断用户是否成功登录过一次
is_login = False # 编辑装饰器
def login(function_name): def for_login(*args, **kwargs):
global is_login # 声明is_login使用的是全局变量 # 如果is_login为True就不需要登录了
if is_login:
res = function_name(*args, **kwargs)
return res # 用户没有登录过的情况
print("你还没有登录哦!")
username = input("请输入用户名:").strip()
password = input("请输入密码:").strip()
if username == 'yume' and password == '123':
print("登录成功!")
is_login = True
res = function_name(*args, **kwargs)
return res
else:
print("用户名或密码错误!") return for_login @login
def print_any():
print("输入什么就输出什么")
s = input()
print(s) @login
def add():
print("将2个字符串相加")
a = input("输入第一个:")
b = input("输入第二个:")
print(a + b) print("""
1.输入什么就输出什么
2.将2个字符串相加
""")
while True:
choice = input("请输入指令:")
if choice == '1':
print_any()
elif choice == '2':
add()
else:
print("错误指令")

python闭包函数与装饰器的更多相关文章

  1. python 闭包函数与装饰器

    1.什么是闭包函数 (1):什么是闭包函数: #内部函数包含对外部作用域而非全局作用域的引用, 简而言之, 闭包的特点就是内部函数引用了外部函数中的变量. 在Python中,支持将函数当做对象使用,也 ...

  2. python闭包函数、装饰器

    闭包函数的传值方式: 方式1:通过参数传值 def func(x): print(x)func(1) 方式2:闭包函数传值 def outter(x): def inner(): print(x) r ...

  3. python闭包函数及装饰器简介

    目录: 闭包函数简介 闭包函数的实际应用 装饰器简介 装饰器初期-完整版 装饰器语法糖 闭包函数简介 1.定义在函数内部的函数(函数的嵌套) 2.内部函数运用外部函数局部名称空间中的变量名 注:函数名 ...

  4. python基础-闭包函数和装饰器

    闭包函数和装饰器 闭包函数 概念:在函数中(嵌套)定义另一个函数时,内部函数引用了外层函数的名字. 特性 闭包函数必须在函数内部定义 闭包函数可引用外层函数的名字 闭包函数是函数嵌套.函数对象.名称空 ...

  5. day11 闭包函数和装饰器

    1.函数不是新知识点 主要是学习函数的用法 在面向对象编程中,一切皆对象,函数也不例外 具体的表现 1.函数可以引用 2.函数可以作为函数的参数 3.函数可以作为函数的返回值 4.可以被存储到容器类型 ...

  6. Python之函数对象、函数嵌套、名称空间与作用域、闭包函数、装饰器

    目录 一 函数对象 二 函数的嵌套 三 名称空间和作用域 四 闭合函数 五 装饰器 一.函数对象 1.函数是第一类对象 #第一类对象的特性:# 可以被引用 # 可以当做参数传递 # 返回值是函数 # ...

  7. 【Python3的命名空间与作用域,闭包函数,装饰器】

    一.命名空间与作用域 命名空间是名字和对象的映射,就像是字典,key是变量名,value是变量的值 1.命名空间的定义 name='egon' #定义变量 def func(): #定义函数 pass ...

  8. day11_7.11 闭包函数与装饰器

    补充: callable 代表可调用的,加括号可以执行.(函数或者类) import this  查看python之禅 一.闭包函数 所谓闭包函数,就是定义在函数内部的函数,也就是函数定义的嵌套.而在 ...

  9. CSIC_716_20191112【闭包函数和装饰器】

    闭包函数 什么是闭包函数:闭包函数是函数嵌套.函数对象.名称空间和作用域的集合体. 闭包函数必须在函数内部定义,闭包函数可以引用外层函数的名字. # _*_ coding: gbk _*_ # @Au ...

随机推荐

  1. 如何0代码实现多人音视频通话?【内附源码/Demo】

    3月15日新增"1860+1194",全国进入了抗疫关键时期.响应政策多地采取了社会面清零策略. 3月14日零点,深圳按下了暂停键. 应疫情防控要求,深圳全市暂停生产经营活动,严格 ...

  2. Flex布局在小程序的使用

    一篇旧文,上手小程序时做的一些探索 Flex布局是一种十分灵活方便的布局方式,目前主流的现代浏览器基本都实现了对Flex布局的完全支持.而在微信小程序中,IOS端使用的渲染引擎WKWebView和安卓 ...

  3. fetch,终于认识你

    fetch和XMLHttpRequest 如果看网上的fetch教程,会首先对比XMLHttpRequest和fetch的优劣,然后引出一堆看了很快会忘记的内容(本人记性不好).因此,我写一篇关于fe ...

  4. wx.getImageInfo和wx.downloadFile下载用户头像报错(小程序canvas以及小程序图片下载部分)

    我先上图 之前我们后台配置的 downloadFile 合法域名是 https://wx.qlogo.cn,   用了好久都没出问题,  前段时间,  用户反馈  分享海报,  用户头像出不来!!!! ...

  5. MySQL中MyISAM和InnoDB引擎的区别

    区别: 1. InnoDB支持事务,MyISAM不支持,对于InnoDB每一条SQL语言都默认封装成事务,自动提交,这样会影响速度,所以最好把多条SQL语言放在begin和commit之间,组成一个事 ...

  6. 微信小程序发送验证码功能,验证码倒计时

    data{ timer:'', countDownNum:'发送验证码', } // 点击验证码倒计时获取验证码 Gain:function(e){ let that = this let count ...

  7. java——封装

    java--封装 java--封装1 封装的理解和好处2 封装的事项实现步骤3 将构造器和setXx结合4 this和super区分 1 封装的理解和好处 隐藏实现细节:[方法(连接数据库)<- ...

  8. 攻防世界——stegano

    分析 1. 一个pdf,里边都是英文. 打开pdf "ctrl + F",检查flag 然活这里边直接告诉你,flag不在这里,一般都这么说了那就是真的不在了. 2. txt打开, ...

  9. MySQL高级以及锁机制

    MySQL高级 推荐阅读: 锁:https://www.cnblogs.com/zwtblog/tag/锁/ 数据库:https://www.cnblogs.com/zwtblog/tag/数据库/ ...

  10. 北桥芯片(north bridge/host bridge)

    看下上面的图,会比较清晰的认识到北桥芯片所在位置 北桥芯片(North Bridge) 是mother board chipset(主板芯片组) 中起主导作用的最重要的组成部分,也称为主桥(Host ...