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

看到没,执行结果一毛一样有木有。


step7:
装饰器用法:
@+函数名
功能:1.自动执行函数(outer函数),并且将下面的函数名f1当做参数
          2.将outer函数的返回值,重新赋值给f1 
需要注意的两点:
1.有return值得才是完整的装饰器。
2.闭包中返回的inner必须是函数的名称func,并非是func(),如果带了括号,那就是执行函数了, inner中返回为None,因为inner中没有return或者return值为其他了。

step8:
有同志就问了,这特么是一个没有参数的,那如果我自己定义的函数有一个参数咋弄。
那我们就用函数中的万能参数*args **kwargs来造一个通用的装饰器吧:
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基础之:九步认识装饰器的更多相关文章

  1. Python菜鸟之路:Python基础-逼格提升利器:装饰器Decorator

    一.装饰器 装饰器是一个很著名的设计模式,经常被用于有切面需求的场景,较为经典的有插入日志.性能测试.事务处理等. 装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量函数中与函数功能本身 ...

  2. python 基础篇 11 函数进阶----装饰器

    11. 前⽅⾼能-装饰器初识本节主要内容:1. 函数名的运⽤, 第⼀类对象2. 闭包3. 装饰器初识 一:函数名的运用: 函数名是一个变量,但他是一个特殊变量,加上括号可以执行函数. ⼆. 闭包什么是 ...

  3. python基础之闭包函数和装饰器

    补充:全局变量声明及局部变量引用 python引用变量的顺序: 当前作用域局部变量->外层作用域变量->当前模块中的全局变量->python内置变量 global关键字用来在函数或其 ...

  4. Python-Day4 Python基础进阶之生成器/迭代器/装饰器/Json & pickle 数据序列化

    一.生成器 通过列表生成式,我们可以直接创建一个列表.但是,受到内存限制,列表容量肯定是有限的.而且,创建一个包含100万个元素的列表,不仅占用很大的存储空间,如果我们仅仅需要访问前面几个元素,那后面 ...

  5. Python基础2:反射、装饰器、JSON,接口

    一.反射 最近接触到python的反射机制,遂记录下来已巩固.但是,笔者也是粗略的使用了__import__, getattr()函数而已.目前,笔者的理解是,反射可以使用户通过自定义输入来导入响应的 ...

  6. Python基础(7)闭包函数、装饰器

    一.闭包函数 闭包函数:1.函数内部定义函数,成为内部函数, 2.改内部函数包含对外部作用域,而不是对全局作用域名字的引用 那么该内部函数成为闭包函数 #最简单的无参闭包函数 def func1() ...

  7. python基础编程: 函数示例、装饰器、模块、内置函数

    目录: 函数示例 装饰器 模块 内置函数 一.函数示例: 1.为什么使用函数之模块化程序设计: 不使用模块程序设计的缺点: 1.体系结构不清晰,可主读性差: 2.可扩展性差: 3.程序冗长: 2.定义 ...

  8. Python基础(闭包函数、装饰器、模块和包)

    闭包函数 格式: def 函数名1(): def 函数名2(): 变量 = 值 return 变量 return 函数名2 func = 函数名1() key = func()

  9. python基础补漏-05-生成器和装饰器

    [1]生成器 很难用简单的语言描述生成器. 生成器:从字面上来理解,就是以某种规则为基础,不断的生成数据的工具 生成器函数: 在函数中如果出现了yield关键字,那么该函数就不再是普通函数,而是生成器 ...

随机推荐

  1. P2161 [SHOI2009]会场预约[线段树/树状数组+二分/STL]

    题目描述 PP大厦有一间空的礼堂,可以为企业或者单位提供会议场地.这些会议中的大多数都需要连续几天的时间(个别的可能只需要一天),不过场地只有一个,所以不同的会议的时间申请不能够冲突.也就是说,前一个 ...

  2. GET和POST的区别【转载】

    GET和POST是HTTP请求的两种基本方法,要说它们的区别,接触过WEB开发的人都能说出一二. 最直观的区别就是GET把参数包含在URL中,POST通过request body传递参数. 你可能自己 ...

  3. 7 html-webpack-plugin的两个基本作用

    html-webpack-plugin的作用: 1.在内存中根据我们的index模板页面,生成一个内存里面的首页 2.当使用html-webpack-plugin之后,我们不再需要手动处理bundle ...

  4. 网站添加logo图片

    网站添加log图片 第一种方法 这里使用的图片一般为16*16大小的图片 <link rel="shortcut icon" href="http://xxx.xx ...

  5. HttpReader

    头文件: #pragma once #include <afxinet.h> class CSF_HttpDataReader { public: CSF_HttpDataReader(v ...

  6. cc 视频的使用

    1. 先上传视频 2.复制代码 3.贴在页面上就可以使用了 4.通过id指定播放那个视频

  7. 智能灯控(基于ZigBee)

    时间:2017年12月 阶段:大二上学期 背景:单片机原理与应用课设 名称:智能灯控 摘要 本系统实现了多方式控灯功能,有按键控灯.串口指令控灯.点对点无线射频控灯.AI模式控灯.其中AI模式控灯是通 ...

  8. [TJOI2015]弦论(第k小子串)

    题意: 对于一个给定的长度为n的字符串,求出它的第k小子串. 有参数t,t为0则表示不同位置的相同子串算作一个,t为1则表示不同位置的相同子串算作多个. 题解: 首先,因为t的原因,后缀数组较难实现, ...

  9. php面向对象之数据隐藏

    什么是数据隐藏? 看到这个有的人会觉得挺不理解的.在前面的文章中,介绍类的时候,我们说定义变量用的关键词是public,但是不止这一个,还有public.private.protected.stati ...

  10. Spring security invalid-session-url 的坑(配了permitAll仍然跳转到登录页)

    Spring security session配置中如果配了如下的invalid-session-url,配置了permitAll链接首次链接系统时会跳转到登录页,将该配置删除即可解决此问题. < ...