一、装饰器:

本质就是函数,功能:为其他函数添加附加功能

原则:

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

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

一个简单的装饰器

import time
def timmer(func):
def wrapper(*args, **kwargs):
start_time = time.time()
res = func(*args, **kwargs)
stop_time = time.time()
print('函数运行的时间%s'%(stop_time - start_time)) return res
return wrapper @timmer
def cal(l):
res = 0
for i in l:
time.sleep(0.01)
res += i
return res
ret = cal(range(200))
print(ret)

二、装饰器的知识储备

装饰器 = 高阶函数 + 函数嵌套 + 闭包

高阶函数定义:

1.函数接收的参数是一个函数名

2.函数的返回值是一个函数名

3.满足上述条件任意一个,都可称之为高阶函数

用高阶函数模拟装饰器:

结果多运行了一次

import time
def foo():
time.sleep(3)
print('来自foo') def timmer(func):
start_time = time.time()
func()
stop_time = time.time()
print('函数运行的时间是%s' %(stop_time - start_time))
return func
foo = timmer(foo)
foo()

  

三、 函数嵌套及闭包

嵌套就是,函数里再定义函数

闭包:闭----->封装的变量   包----->一层层函数

每个包找变量先从自身那层开始找,找不到再从外层找

重点:函数即变量

def father(name):

    def son():
print('我的爸爸是%s' %name)
def grandson():
name = 'China'
print('我的爷爷是%s' %name)
grandson()
son()
father('河南')

装饰器框架

def timmer(func):
def wrapper():
#增加功能
func() return wrapper

补充该装饰器架子

import time
def timmer(func):
def wrapper():
#增加功能
start_time = time.time()
func()
stop_time = time.time()
print('运行时间是%s'%(stop_time - start_time))
return wrapper def test():
time.sleep(2)
print('test函数运行完毕') res = timmer(test) #返回的是wrapper的地址
res() #执行的是wrapper()

可以看出,只要把函数的最后两行代码的函数接收值改为:test

test= timmer(test)

test() 

看起来符合装饰器原则

但每次给函数增加功能都得写一遍 test= timmer(test),显然不合理

所以,用@timmer 去代替test = timmer(test)

import time
def timmer(func):
def wrapper():
#增加功能
start_time = time.time()
func()
stop_time = time.time()
print('运行时间是%s'%(stop_time - start_time))
return wrapper
@timmer #就相当于:test = timmer(test)
def test():
time.sleep(2)
print('test函数运行完毕')
test()

四、加上返回值的装饰器

import time
def timmer(func):
def wrapper():
#增加功能
start_time = time.time()
res = func() #这里执行的函数test,所以要在这里给函数一个接收返回值的变量
stop_time = time.time()
print('运行时间是%s'%(stop_time - start_time))
return res #这里要返回test的返回值
return wrapper
@timmer #就相当于:test = timmer(test)
def test():
time.sleep(2)
print('test函数运行完毕')
return '这是test的返回值'
a = test()
print(a)

五、加上参数的装饰器

import time
def timmer(func):
def wrapper(*args, **kwargs): #可接受任意值
#增加功能
start_time = time.time()
res = func(*args, **kwargs) #将wrapper接受的值原封传给func
stop_time = time.time()
print('运行时间是%s'%(stop_time - start_time))
return res
return wrapper
@timmer #就相当于:test = timmer(test)
def test1(name, age, gander):
time.sleep(2)
print('test1运行结果名字为%s年龄为%d性别为%s'%(name, age, gander))
return '这是test1的返回值'
a = test1('jinling', 20, 'female')
print(a) @timmer
def test2(name, age):
time.sleep(3)
print('test2函数运行结果名字为%s年龄为%d'%(name, age))
return '这是test2的返回值'
b = test2('liuwen', 18)
print(b)

六、实现一个验证功能装饰器

def func_ver(func):
def wrapper(*args, **kwargs):
name1 = input('请输入用户名:')
pd = input('请输入用户密码:')
if name1 == 'PJL' and pd == '123':
func(*args, **kwargs)
else:
print('用户名或密码错误') return wrapper @func_ver
def index():
print('欢迎来到的我的网站')
@func_ver
def home(name):
print('%s欢迎回家'%name)
@func_ver
def secrect(name):
print('%s这是你的小秘密'%name) index()
home('jinling')
secrect('jining')

 

 

  

最新文章

  1. Functional Programming without Lambda - Part 2 Lifting, Functor, Monad
  2. 随感一:android handler传值更改ui
  3. Android——ScrollView
  4. 平衡工作与生活的艺术——GTD简单介绍
  5. NFS配置及使用
  6. 【转载】DQL、DML、DDL、DCL的概念与区别
  7. Python3.7版本unittest框架添加用例的方法
  8. 【CF768G】The Winds of Winter 可持久化线段树 DFS序
  9. Windows10 VS2017 C++ xml解析(tinyxml2库)
  10. 在WINDOWS7下无U盘安装WINDOWS10
  11. c# 取本地ip地址
  12. 点击复制内容到剪切板(clipboard)
  13. 实例化Flask的参数 及 对app的配置
  14. (转)typedef 函数指针的用法
  15. HOJ 1936&POJ 2955 Brackets(区间DP)
  16. 【linux】文件目录说明
  17. 2016-2017-2 20155223 实验二 《Java面向对象程序设计》
  18. Emacs ^ Vim
  19. 使用C#委托来实现异步编程
  20. hibernate.hbm2ddl.auto=update不能自动生成表结构

热门文章

  1. ThreadLocal使用原理、注意问题、使用场景
  2. 微信小程序商城 带java后台源码
  3. 跟我学SpringCloud | 第三篇:服务的提供与Feign调用
  4. C# RESTful API 访问辅助类
  5. 系统学习 Java IO (十二)----数据流和对象流
  6. 点菜网---Java开源生鲜电商平台-商品基础业务架构设计-商品分类(源码可下载)
  7. Java虚拟机详解(二)------运行时内存结构
  8. docker 获取镜像
  9. Python笔记【4】_字典学习
  10. [apue] 测试管道容量的一些疑问