装饰器前奏:

一.定义:

  1.装饰器本质是函数,语法都是用def去定义的

  (函数的目的:他需要完成特定的功能)

  2.装饰器的功能:就是装饰其他函数(就是为其他函数添加附加功能)

二.原则:

  1. 不能修改被装饰的函数的源代码

  2. 不能修改被修饰的函数的调用方式

三. 实现装饰器知识储备:

  1.函数即“变量”  (先定义,再调用)

  2.高阶函数

  3.嵌套函数

  4.匿名函数   样式 (calc = lambde x:x*3 )

高阶函数 + 嵌套函数 =》 装饰器

1. 函数即“变量”

  (1). python的内存机制

1
2
3
4
5
#变量
= 1
#函数
def test():
    pass

     以上一个变量一个函数在内存中的表现形式如下图:

     在python解释器中,有一个概念叫做引用基数,那什么叫引用基数呐,就是比方说,x=1,它会先在内存当中把1这个值试试在在的存放下来,这个x其实就是1的门牌号,也是对1的一次引用。python什么时候把这个1这个屋子清空呐?它会等到1所对应的门牌号都没有了,就会把1这里面的东西给清掉,这个也是python的内存回收机制,就是靠这种方式回收的。

  (2). del清理

    那我们用什么清理呐?用del去清理门牌号,就是对1的值引用的变量,del  x就表示清理掉1对应的x的门牌号。如果x没有被del,则x永远不还被删除,除非程序结束了,不然永远不会被删除。del删除的不是1,只是把门牌号x删除了,只是定期刷新时,发现1没有被其他门牌号引用了,才会被清掉。

  (3). 函数在内存的表现形式

  我们先通过四个例子来解释一下:

  ①bar函数没有定义

1
2
3
4
5
6
7
8
9
10
11
12
 
#bar函数没有定义
def foo():
    print("in the foo")
    bar()
 
foo()
 
#输出
in the foo

NameError: name 'bar' is not defined

 

  ②bar函数在foo函数之后定义

1
2
3
4
5
6
7
8
9
10
11
12
13
#bar函数在foo函数之后定义
def foo():
    print("in the foo")
    bar()
 
def bar():
    print("in the bar")
 
foo()
 
#输出
in the foo
in the bar

 

 ③bar函数是在foo函数之前定义

1
2
3
4
5
6
7
8
9
10
11
12
13
# bar函数是在foo函数之前定义
def bar():
    print("in the bar")
 
def foo():
    print("in the foo")
    bar()
 
foo()
 
#输出
in the foo
in the bar

  显然,两种写法效果是一样的,那我们来看看第四种情况。

  ④bar函数在foo函数调用之后声明

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
# bar函数在foo函数调用之后声明
def foo():
    print("in the foo")
    bar()
 
foo()
 
def bar():
    print("in the bar")
 
#输出
Traceback (most recent call last):
in the foo
  File "D:/PycharmProjects/pyhomework/day4/装饰器/函数即变量.py", line 31in <module>
    foo()
  File "D:/PycharmProjects/pyhomework/day4/装饰器/函数即变量.py", line 29in foo
    bar()
NameError: name 'bar' is not defined  #bar函数没有定义

2.高阶函数

满足下面条件之一就可以称为高阶函数:

  a.把一个函数名当做实参传给另一个函数

  b.返回值中包含函数名(调用函数体)

1、把一个函数名当做实参传给另外一个函数

作用:在不修改被装饰函数源代码的情况下为其添加功能

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
def bar():
    time.sleep(3)
    print("in the bar")
 
def test1(func):
    print(func)
    start_time = time.time()
    func()   #run bar
    stop_time = time.time()
    print("the func run the is %s"%(stop_time-start_time))
#没有修改bar的代码
test1(bar)  #把bar函数名当做实参传到test1中
 
#输出
<function bar at 0x0000000000A7D378>  #bar函数的内存地址
in the bar
the func run the is 2.9912972450256348

2、返回值中包括函数名

作用:不修改函数调用方式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
import  time
 
def bar():
    time.sleep(3)
    print("in the bar")
 
def test2(func):
    print(func)
    return func   #返回函数的内存地址
 
#调用test2函数
bar = test2(bar) 
bar()  #没有bar函数改变调用方式
 
#输出
<function bar at 0x0000000000B6D378>  #打印bar函数的内存地址
in the bar
 

小白的Python之路 day4 装饰器前奏的更多相关文章

  1. 小白的Python之路 day4 装饰器高潮

    首先装饰器实现的条件: 高阶函数+嵌套函数 =>装饰器 1.首先,我们先定义一个高级函数,去装饰test1函数,得不到我们想要的操作方式 import time #定义高阶函数 def deco ...

  2. python之路之装饰器

    一 装饰器进化之路1) import time def index(): start_time=time.time() time.sleep() print('welcome to index wor ...

  3. [Python之路] 使用装饰器给Web框架添加路由功能(静态、动态、伪静态URL)

    一.观察以下代码 以下来自 Python实现简易HTTP服务器与MINI WEB框架(利用WSGI实现服务器与框架解耦) 中的mini_frame最后版本的代码: import time def in ...

  4. 小白的Python之路 day4 生成器

    一.列表生成式  看下面例子: 列表生成式的作用:主要是让代码更简洁(还有装X的效果) 二.生成器 通过列表生成式,我们可以直接创建一个列表.但是,受到内存限制,列表容量肯定是有限的.而且,创建一个包 ...

  5. 小白的Python之路 day4 迭代器

    迭代器 学习前,我们回想一下可以直接作用于for循环的数据类型有以下几种: 1.集合数据类型,如list.tuple.dict.set.str等: 2.是generator,包括生成器和带yield的 ...

  6. 小白的Python之路 day4 json and pickle数据标准序列化

    一.简述 我们在写入文件中的数据,只能是字符串或者二进制,但是要传入文件的数据不一定全是字符串或者二进制,那还要进行繁琐的转换,然后再读取的时候,还要再转回去,显得很麻烦,今天就来学习标准的序列化:j ...

  7. 小白的Python之路 day4 软件目录结构规范

    软件目录结构规范 为什么要设计好目录结构? "设计项目目录结构",就和"代码编码风格"一样,属于个人风格问题.对于这种风格上的规范,一直都存在两种态度: 一类同 ...

  8. 小白的Python之路 day4 不同目录间进行模块调用(绝对路径和相对路径)

    一.常用模块调用函数功能解释 1.__file__ 功能:返回自身文件的相对路径 你从pycharm的执行结果可以看出,在pycharm执行atm.py文件时,是从绝对路径下去执行的,而你从cmd下去 ...

  9. 小白的Python之路 day4 生成器并行运算

    一.概述 我们已经明白生成器内部的结构,其实就是通过像函数这样的东西实现的! 多线程和单线程:简单来说多线程就是并行运算,单线程就是串行运算 二.生成器执行原理 第一步:生成一个生成器  第二步:执行 ...

随机推荐

  1. 在64位Win7环境+64位JDK下,运行64位Eclipse,提示“Failed to load the JNI shared library”错误,提示jvm.dll不对

    -startup plugins/org.eclipse.equinox.launcher_1.2.0.v20110502.jar --launcher.library plugins/org.ecl ...

  2. js的数组方法整理

    slice 从已有的数组中返回选定的元素.该方法不会修改数组,而是返回一个子数组. 语法:arr.slice(start,end) start: 必须,规定从何处开始选取.如果是负数,就是从尾部开始算 ...

  3. T-SQL的进阶:超越基本级别3:构建相关子查询——701小组

    T-SQL的进阶:超越基本级别3:构建相关子查询 格雷戈里·拉森,2014/03/05 原文链接: http://www.sqlservercentral.com/articles/Stairway+ ...

  4. 在LwIP 协议栈移植 Snap 7

    本文欢迎引用,转载. 引用,转载请标明出处! 调试完毕源码将上传到GitHub 为了嵌入式系统与STEP 7 PLC 通过S7 协议通讯,尝试移植 Snap 7 到STM32F407 cpu 上. 今 ...

  5. Django框架中的视图和模板

    视图views django中的视图就是用来定义函数来处理一些逻辑的核心地方. django中通过urls来建立路径跟views中的视图函数的映射关系. urls中的映射关系 ''' urlpatte ...

  6. SpringMVC集成Shiro、读取数据库操作权限

    1.Maven添加Shiro所需的jar包 <dependency> <groupId>org.apache.shiro</groupId> <artifac ...

  7. ##5.2 Nova计算节点-- openstack pike

    ##5.2 Nova计算节点 openstack pike 安装 目录汇总 http://www.cnblogs.com/elvi/p/7613861.html ##5.2 Nova计算节点 # co ...

  8. Android OpenGL ES 开发(三): OpenGL ES 定义形状

    在上篇文章,我们能够配置好基本的Android OpenGL 使用的环境.但是如果我们不了解OpenGL ES如何定义图像的一些基本知识就使用OpenGL ES进行绘图还是有点棘手的.所以能够在Ope ...

  9. Retrofit网络请求库应用02——json解析

    PS:上一篇写了Retrofit网络请求库的简单使用,仅仅是获取百度的源码,来证明连接成功,这篇讲解如何解析JSON数据,该框架不再是我们之前自己写的那样用JsonArray等来解析,这些东西,我们都 ...

  10. 构建Nginx均衡LAMP高性能服务器

    LNAMP(Linux+Nginx+Apache+Mysql+PHP)架构受到很多IT企业的青睐,取代了原来认为很好的LNMP(Linux+Nginx+Mysql+PHP)架构,那我们说LNAMP到底 ...