一、什么是装饰器

当我们做好一个产品之后,需要对它进行不断地维护,对某些函数增加一些功能。这个时候如果去修改源代码将是非常不合适的。(原因:1.原则上已经写好的函数尽量不去修改它,因为一旦修改可能会导致不可预知的错误发生或者降低稳定性。2.函数可能被调用很多很多次,如果修改函数有可能会导致调用方式发生改变,会有大量的修改工作。)所以,装饰器就出现了。它可以做到为函数增添新功能的同时而不修改函数本身的源代码。

二、装饰器的原则

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

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

三、实现装饰器的知识储备

1.函数即“变量”:定义一个函数,相当于将函数体赋值给函数名这个变量。

2.高阶函数:满足以下两个条件之一的函数即为高阶函数:①把函数名当做实参传给另一个函数②返回值中包含函数名

3.嵌套函数:函数是可以嵌套的

四、使用装饰器

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

装饰器的本质是函数,功能是为其它函数添加附加功能

#现在有两个函数,分别有各自的功能,需求是计算每个函数运行的时间

def text1():
time.sleep(2)
print("this is text1") def text2(*args,**kwargs):
time.sleep(2)
print("this is text2",args[0],args[1])

可以写装饰器:

def timer(func):
def deco(*args,**kwargs):
start_time=time.time()
func(*args,**kwargs)
end_time=time.time()
print("the function tun time is {0}".format(end_time-start_time))
return deco

其中func()函数是原函数真正的执行过程,deco函数内其他的语句都是新添加功能的语句

装饰器名叫timer

写好装饰器就行了吗?不是,还差一个步骤,在原函数上方增加一条语句,整合到一起:

def timer(func):
def deco(*args,**kwargs):
start_time=time.time()
func(*args,**kwargs)
end_time=time.time()
print("the function tun time is {0}".format(end_time-start_time))
return deco @timer #此操作的作用是将装饰器的功能加到原函数上 相当于text1=timer(text1)
def text1():
time.sleep(2)
print("this is text1") @timer #没有@timer语句则不会将装饰器功能加上
def text2(*args,**kwargs): #如果原函数有参数,需要在装饰器中deco上加参数
time.sleep(2)
print("this is text2",args[0],args[1]) text1() #不改变调用方式,且增加了新功能
text2(29000,"hello")

装饰器解析:deco函数用来实现新增功能,timer函数和@timer语句用来使调用方式不发生改变

执行结果:

五、带参数的装饰器

这里说的不是原函数带参数,而是装饰器函数本身带参数

现在有需求,用装饰器实现登陆过程,并且有两种不同的登陆方式:使用本地数据登陆和使用云端数据登陆

#现在有三个网页,其中登陆home时用本地数据,登陆bbs时用云端数据

def index():
print("welcome to index page") def home():
print("welcome to home page")
return "form home" def bbs():
print("welcome to bbs page")

需要装饰器:

name="alex"
word=""
def auth(auth_type):
def auth_out(func):
def deco(*args,**kwargs):
if auth_type=="local": #如果使用本地数据
uesrname=input("请输入用户名:").strip()
password=input("请输入密码:").strip()
if uesrname==name : # 如果用户存在
if password==word:#如果密码正确
print("使用本地数据")
print("欢迎进入!")
return func(*args,**kwargs) #进入home界面
else:
print("密码错误")
else:
print("用户不存在")
elif auth_type=="cloude": #如果使用云端数据
print("使用云端数据")
print("欢迎")
func() # 进入bbs界面
return deco
return auth_out def index():
print("welcome to index page") @auth(auth_type="local") #登录home界面使用本地用户数据,在auth()内传参
def home():
print("welcome to home page")
return "form home" @auth(auth_type="cloude") #登录bbs界面使用云端用户数据
def bbs():
print("welcome to bbs page") index()
print(home())
bbs()

其实就是将原来的装饰器外层又嵌套了一层函数,用来传递参数。@语句也变成了@auth(auth_type=" ")

运行结果:

初学Python——装饰器的更多相关文章

  1. python 装饰器学习(decorator)

    最近看到有个装饰器的例子,没看懂, #!/usr/bin/python class decorator(object): def __init__(self,f): print "initi ...

  2. 关于python装饰器

    关于python装饰器,不是系统的介绍,只是说一下某些问题 1 首先了解变量作用于非常重要 2 其次要了解闭包 def logger(func): def inner(*args, **kwargs) ...

  3. python装饰器通俗易懂的解释!

    1.python装饰器 刚刚接触python的装饰器,简直懵逼了,直接不懂什么意思啊有木有,自己都忘了走了多少遍Debug,查了多少遍资料,猜有点点开始明白了.总结了一下解释得比较好的,通俗易懂的来说 ...

  4. Python 装饰器学习

    Python装饰器学习(九步入门)   这是在Python学习小组上介绍的内容,现学现卖.多练习是好的学习方式. 第一步:最简单的函数,准备附加额外功能 1 2 3 4 5 6 7 8 # -*- c ...

  5. python 装饰器修改调整函数参数

    简单记录一下利用python装饰器来调整函数的方法.现在有个需求:参数line范围为1-16,要求把9-16的范围转化为1-8,即9对应1,10对应2,...,16对应8. 下面是例子: def fo ...

  6. Python装饰器详解

    python中的装饰器是一个用得非常多的东西,我们可以把一些特定的方法.通用的方法写成一个个装饰器,这就为调用这些方法提供一个非常大的便利,如此提高我们代码的可读性以及简洁性,以及可扩展性. 在学习p ...

  7. 关于python装饰器(Decorators)最底层理解的一句话

    一个decorator只是一个带有一个函数作为参数并返回一个替换函数的闭包. http://www.xxx.com/html/2016/pythonhexinbiancheng_0718/1044.h ...

  8. Python装饰器由浅入深

    装饰器的功能在很多语言中都有,名字也不尽相同,其实它体现的是一种设计模式,强调的是开放封闭原则,更多的用于后期功能升级而不是编写新的代码.装饰器不光能装饰函数,也能装饰其他的对象,比如类,但通常,我们 ...

  9. Python装饰器与面向切面编程

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

随机推荐

  1. CSS实现两列布局,一列固定宽度,一列宽度自适应方法

    不管是左是右,反正就是一边宽度固定,一边宽度自适应. 博客园的很多主题也是这样设计的,我的博客也是右侧固定宽度,左侧自适应屏幕的布局方式. html代码: <div id="wrap& ...

  2. RabbitMQ 环境搭建

    安装基础环境 yum install net-tools yum install yum yum install gcc glibc-devel make ncurses-devel openssl- ...

  3. <自动化测试方案_6>第六章、API自动化测试

    第六章.API自动化测试 (一)工具实现 目前大众接口测试的工具有:Postman.SoupUI.jmeter他们的特点介绍有人做个宏观的研究,这里进行引用:https://blog.csdn.net ...

  4. mysql数据库的基本操作:创建数据库、查看数据库、修改数据库、删除数据库

    本节相关: 创建数据库 查看数据库 修改数据库 删除数据库 首发时间:2018-02-13 20:47 修改: 2018-04-07:考虑到规范化,将所有语法中“关键字”变成大写;以及因为整理“mys ...

  5. linux富规则的案例

    使用firewall-cmd添加富规则: 1.拒绝来自public区域中ip地址为192.168.0.11的所有流量 firewall-cmd --permanent --zone=public -- ...

  6. [20170607]再论Private Strand Flush Not Complete.txt

    [20170607]再论Private Strand Flush Not Complete.txt --//曾经跟zergduan有过讨论,链接www.itpub.net/thread-2065703 ...

  7. web前端(14)—— JavaScript的数据类型,语法规范1

    编辑器选择 对js的编辑器选用,有很多,能对html编辑的,也能对js编辑,比如notepad++,visual studio code,webstom,atom,pycharm,sublime te ...

  8. 在Linux系统上利用Tomcat搭建测试环境

    第一歩:查看Linux系统的IP地址. 输入命令:ifconfig 第二歩:WinSCP工具 1.下载WinSCP工具,便于文件直接从windows系统直接拖动到Linux系统中,图形化创建文件夹等. ...

  9. php学习----文件系统

    PHP文件系统之读取文件内容 PHP具有丰富的文件操作函数,最简单的读取文件的函数为file_get_contents,可以将整个文件全部读取到一个字符串中. $content = file_get_ ...

  10. 注入攻击(SQL注入)

    注入攻击是web安全领域中一种最为常见的攻击方式.注入攻击的本质,就是把用户输入的数据当做代码执行.这里有两个关键条件,第一是用户能够控制输入,第二个就是原本程序要执行的代码,将用户输入的数据进行了拼 ...