python基础之:九步认识装饰器
step1.
先看个代码吧:
def f():
print('') f=lambda a:a +100 #覆盖上面的函数f print(f) #函数名指函数所在内存中的位置,入带后面括号表示执行函数
print(f(10)) out:
<function <lambda> at 0x101b7b6a8>
110
以上code说明:
1.在def 函数时,以顺序执行,如果相同的函数名,会被最后函数覆盖以前的
2.如果直接func名而没有后面的圆括号的话,只指向函数在内存中的位置
3.lambda表达式会自动return结果,而def需要定义return值
step2:
python中可想函数传递参数,并将参数转变为本地变量存在于函数内部。
def f(x):
print(locals()) f('a') out: {'x': 'a'}
step3:
python函数中可以嵌套函数,这就说明了我们可以在函数里定义函数,而且现有的作用域和函数的变量生存周期依旧适用。
def outer():
x='hello world'
def inner():
print(x) #
inner() # outer() out:
hello world
#1 中发生了什么,函数inner需要一个x的变量,去查找x的变量,但本地变量中没有,查找失败后去上一层的作用域中去寻找,而x变量在上一层函数的作用域中。
对outer来说,x是本地变量,但函数inner可以访问封闭的作用域
#2 中,调用了inner函数,python解释器会优先查找本地的变量x
step4:
def outer():
x='hello world'
def inner():
print(x)
return inner #1 foo=outer() #2
print(foo)
foo() out:
<function outer.<locals>.inner at 0x10137b6a8>
hello world
#1与step3区别,我们将函数inner的内存地址给返回了
#2,将函数outer的返回值赋值于变量foo,也就是说,foo() 执行时,实际上执行了inner()。
但inner函数集成了outer函数中的本地变量,outer外面,没有x这个变量,为什么能执行呢??
这是因为python中有一个叫”函数闭包”的特性,怎么理解呢?嵌套定义在非全局作用域中的函数,能够记住它在被定义时候它所处的封闭命名空间。
函数outer每次在被调用的时候,函数inner都会被重新定义,现在变量x的值不会变化,所以每次执行的逻辑和结果都一样,那么如果x是变化的呢?
step5:
def outer(x):
def inner():
print(x) #1
return inner print1=outer(1)
print2=outer(2) print1()
print2() out:
1
2
从这个我们能够看到“闭包”:被函数记住的封闭作用域,能够被用来创建自定义的函数。
事实上,我们并不是传递参数1或2给inner函数,我们实际上是定义了能够打印各种数字的自定义版本。
step6:装饰器
终于说到装饰器上了。。。。。
装饰器其实就是一个闭包而已,把func作为参数,然后返回一个替代版的函数,其实真正执行的是inner内部的流程
def outer(func):
def inner():
print('before some_func')
res = func()
print(res+1)
print(('after some_func'))
return res+1
return inner
def foo():
return 1
f1=outer(foo) #1
f1() out:
before some_func
2
after some_func
在这我们可以认为变量f1是函数foo的的装饰器。但这么写有点太low了是不是,所以python中有个优雅的@符号,太装逼了。。。可以这么写:
def outer(func):
def inner():
print('before some_func')
res = func()
print(res+1)
print(('after some_func'))
return res+1
return inner
@outer
def foo():
return 1
foo() out: before some_func
2
after some_func
看到没,执行结果一毛一样有木有。
step8:
def outer(func):
def inner(*args,**kwargs):
print('before some_func')
res = func(*args,**kwargs)
print(('after some_func'))
return res
return inner
@outer
def foo():
print(111)
foo() @outer
def foo1(li,d):
print('args:%s'%li)
print('kwargs:%s'%d)
foo1('fuck','U')
===============
out:
before some_func
111
after some_func
before some_func
args:fuck
kwargs:U
after some_func
step9:
多层装饰器
装饰器类似于俄罗斯套娃,可以一层套一层,譬如:
def outer(func):
def inner(*args,**kwargs):
print('before some_func')
res = func(*args,**kwargs)
print(('after some_func'))
return res
return inner @outer1
@outer
def foo1(li,d):
print('args:%s'%li)
print('kwargs:%s'%d)
foo1('fuck','U') ====================
out:
before:this is outer1
before some_func
args:fuck
kwargs:U
after some_func
after:this is outer1 again!
未完待续。。。
python基础之:九步认识装饰器的更多相关文章
- Python菜鸟之路:Python基础-逼格提升利器:装饰器Decorator
一.装饰器 装饰器是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理等. 装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量函数中与函数功能本身 ...
- python 基础篇 11 函数进阶----装饰器
11. 前⽅⾼能-装饰器初识本节主要内容:1. 函数名的运⽤, 第⼀类对象2. 闭包3. 装饰器初识 一:函数名的运用: 函数名是一个变量,但他是一个特殊变量,加上括号可以执行函数. ⼆. 闭包什么是 ...
- python基础之闭包函数和装饰器
补充:全局变量声明及局部变量引用 python引用变量的顺序: 当前作用域局部变量->外层作用域变量->当前模块中的全局变量->python内置变量 global关键字用来在函数或其 ...
- Python-Day4 Python基础进阶之生成器/迭代器/装饰器/Json & pickle 数据序列化
一.生成器 通过列表生成式,我们可以直接创建一个列表.但是,受到内存限制,列表容量肯定是有限的.而且,创建一个包含100万个元素的列表,不仅占用很大的存储空间,如果我们仅仅需要访问前面几个元素,那后面 ...
- Python基础2:反射、装饰器、JSON,接口
一.反射 最近接触到python的反射机制,遂记录下来已巩固.但是,笔者也是粗略的使用了__import__, getattr()函数而已.目前,笔者的理解是,反射可以使用户通过自定义输入来导入响应的 ...
- Python基础(7)闭包函数、装饰器
一.闭包函数 闭包函数:1.函数内部定义函数,成为内部函数, 2.改内部函数包含对外部作用域,而不是对全局作用域名字的引用 那么该内部函数成为闭包函数 #最简单的无参闭包函数 def func1() ...
- python基础编程: 函数示例、装饰器、模块、内置函数
目录: 函数示例 装饰器 模块 内置函数 一.函数示例: 1.为什么使用函数之模块化程序设计: 不使用模块程序设计的缺点: 1.体系结构不清晰,可主读性差: 2.可扩展性差: 3.程序冗长: 2.定义 ...
- Python基础(闭包函数、装饰器、模块和包)
闭包函数 格式: def 函数名1(): def 函数名2(): 变量 = 值 return 变量 return 函数名2 func = 函数名1() key = func()
- python基础补漏-05-生成器和装饰器
[1]生成器 很难用简单的语言描述生成器. 生成器:从字面上来理解,就是以某种规则为基础,不断的生成数据的工具 生成器函数: 在函数中如果出现了yield关键字,那么该函数就不再是普通函数,而是生成器 ...
随机推荐
- 匿名函数、sorted()、filter()、map()、递归
一.匿名函数 1.lambda 匿名函数 方法 lambda 参数:返回值 (函数名统一叫lambda) def func(n): return n**2 print(func(3)) #这是一个普通 ...
- flask中使用ajax 处理前端请求 弹框展示
菜小鱼初次使用 ajax,想前端提交数据,后端处理后,将结果以弹框的形式展示,在网上查看了好多,不停的调试,终于调通了 html: <html> <head></head ...
- linux服务器上安装jenkins
nkins常用的有两种安装方式: 1.直接下载war包jenkins.war,下载地址https://jenkins.io/download 直接下载 1.1.可以把war包直接部署到servlet容 ...
- sqlserver内存、会话、连接查询
1.连接查询 select * from sysprocesses where dbid in (select dbid from sysdatabases where name='dbname') ...
- PHP爬虫之queryList
根据queryList 自己花了一个下午的时间写了一个爬星座数据的类,完全手写.附上代码 <?php require '../vendor/autoload.php'; use QL\Query ...
- Vagrant Docker Composer Yarn 国外资源下载慢或失败的问题
1 问题 有时,我们请求国外资源时,下载巨慢,甚至失败.如: cd vue-devtools/ $ yarn install 进行到 cypress.... 时,可能失败. 2 解决 次日凌晨(7-8 ...
- Java上传大文件夹
javaweb上传文件 上传文件的jsp中的部分 上传文件同样可以使用form表单向后端发请求,也可以使用 ajax向后端发请求 1.通过form表单向后端发送请求 <form id=" ...
- NetworkX系列教程(8)-Drawing Graph
小书匠Graph图论 如果只是简单使用nx.draw,是无法定制出自己需要的graph,并且这样的graph内的点坐标的不定的,运行一次变一次,实际中一般是要求固定的位置,这就需要到布局的概念了.详细 ...
- (转)实验文档5:企业级kubernetes容器云自动化运维平台
部署对象式存储minio 运维主机HDSS7-200.host.com上: 准备docker镜像 镜像下载地址 复制 12345678910111213141516 [root@hdss7-200 ~ ...
- springboot连接redis进行CRUD
springboot连接redis进行CRUD: 1.添加以下依赖: <dependency> <groupId>org.springframework.boot</gr ...