Python装饰器

装饰器是在不修改源码给代码添加功能的常用方法。@是装饰的标志。我们知道,在给代码增加功能的时候,要遵循开放封闭的原则,不能随便更改原码,因此装饰器的功能就显示出来了,只需要在函数前面加上装饰器就能解决问题所需。

def verification(func):
#验证模块
def inner():
print("请输入你的账号和密码进行验证")
func()
return inner def f1():
print() def f2():
print() def f3():
print() def f4():
print() f2 = verification(f2)
print(f2)
f2()

上面代码中,我们给f2增加了一个验证功能,很多时候,我们需要在不改变原码的情况下对程序的功能进行扩展。上面是装饰器的原理。装饰器相当于在函数上面添加了一句代码f = func(f),让程序在执行的时候能够先进行验证,下面来看一下完整的装饰器。

def verification(func):
#验证模块
def inner():
print("请输入你的账号和密码进行验证")
func()
return inner def f1():
print() def f2():
print() def f3():
print() def f4():
print() f1 = verification(f1)
f2 = verification(f2)
f3 = verification(f3)
f4 = verification(f4) f2() #此处f2即inner的函数地址,执行f2()其实执行的是inner()

上述代码中,我们增加了四局代码,使我们能够在不修改原码的情况下,对程序进行了扩充,并且没有改变其他部门使用人的习惯,很好的解决了问题,这种方法其实就是装饰器的原理。我们进行简单的修改就能够更加好看和美观。


def verification(func):
#验证模块
def inner():
print("请输入你的账号和密码进行验证")
func()
return inner @verification #f1 = verification(f1)
def f1():
print(666) @verification #等价于f2 = verification(f2)
def f2():
print(777) @verification #f3 = verification(f3)
def f3():
print(888) @verification #f4 = verification(f4)
def f4():
print(999) f2() #此处f2即inner的函数地址,执行f2()其实执行的是inner()
 

上面代码中,我们应用了装饰器@,也实现了与上面代码一样的功能,其实函数f1上面的装饰器@verification等价于f1 = verification(f1)这句话,让我们在进行函数之前,首先执行验证代码块,由于verification(f1)是执行验证函数,此时用户还没有调用,但是函数已经执行,为了不让函数执行验证模块,我们增加了一层函数嵌套,在第二层进行嵌套,第一层的作用是返回第二层嵌套函数的函数名,以便在用户输入调用的时候在进行调用,所以用户在调用的时候,输入f2()其实等于执行了函数inner()。此时inner()函数也激活了,在python内存中存在。

因此,如果我们不想某层函数执行的话,就给函数嵌套一层函数,返回第一层函数的函数名,在第二层进行执行。我们也可以给函数增加参数,要知道是那个函数在执行命令。

给函数传递参数,给函数传递一个参数,由于我们知道,用户调用f2()的时候,其实运行的是inner(),因此inner()的参数与func()一样多。下面我们来给装饰器传递一个参数。

def verification(func):
#验证模块
def inner(name):
print("请输入你的账号和密码进行验证")
func(name)
return inner @verification #f1 = verification(f1)
def f1(name):
print() @verification #等价于f2 = verification(f2)
def f2(name):
print(,name) @verification #f3 = verification(f3)
def f3(name):
print() @verification #f4 = verification(f4)
def f4():
print() f2("alex") #此处f2即inner的函数地址,执行f2()其实执行的是inner()

传递多组参数:

def verification(func):
#验证模块
def inner(*args,**kwargs):
print("请输入你的账号和密码进行验证")
func(*args,**kwargs)
return inner @verification #f1 = verification(f1)
def f1(name):
print(666) @verification #等价于f2 = verification(f2)
def f2(name):
print(777,name) @verification #f3 = verification(f3)
def f3(name,age,*args):
print(888,name,age,args) @verification #f4 = verification(f4)
def f4():
print(999) f2("alex") #此处f2即inner的函数地址,执行f2()其实执行的是inner()
f3("alex","sb","is",333,222)

我们结合之前学习的动态参数*args,**kwargs可以给函数传递多个参数,这样能满足用户传入多个参数不会出错的情况。

上面代码运行结果如下:

请输入你的账号和密码进行验证
  777 alex
  请输入你的账号和密码进行验证
  888 alex sb ('is', 333, 222)
    我们给f3()传递了多个参数,但是系统并没有报错,而且正常接受了参数,可见使用*args,**kwargs能够接收多个参数,并且不同的函数即便函数的参数的个数不一致也没有关系。

装饰器就是为了给已经存在的函数扩展新的功能。

day4 装饰器深入解析的更多相关文章

  1. 小白的Python之路 day4 装饰器前奏

    装饰器前奏: 一.定义: 1.装饰器本质是函数,语法都是用def去定义的 (函数的目的:他需要完成特定的功能) 2.装饰器的功能:就是装饰其他函数(就是为其他函数添加附加功能) 二.原则: 1. 不能 ...

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

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

  3. Day4 装饰器——迭代器——生成器

    一 装饰器 1.1 函数对象 一 函数是第一类对象,即函数可以当作数据传递 #1 可以被引用 #2 可以当作参数传递 #3 返回值可以是函数 #3 可以当作容器类型的元素 二 利用该特性,优雅的取代多 ...

  4. day4装饰器

    Python装饰器 1.必备 def foo(): print(foo) <function foo at 0x7f62db093f28> >>> foo <fun ...

  5. day4装饰器-迭代器&&生成器

    一.装饰器 定义:本质是函数,(装饰其他函数)就是为其它函数添加附加功能 原则:1.不能修改被装饰的函数的源代码 2.不能修改被装饰的函数的调用方式 实现装饰器知识储备: 1.函数及“变量” 2.高阶 ...

  6. 装饰器、生成器,迭代器、Json & pickle 数据序列化

    1. 列表生成器:代码例子 a=[i*2 for i in range(10)] print(a) 运行效果如下: D:\python35\python.exe D:/python培训/s14/day ...

  7. 装饰器模式&&ES7 Decorator 装饰器

    装饰器模式(Decorator Pattern)允许向一个现有的对象动态添加新的功能,同时又不改变其结构.相比JavaScript中通过鸡肋的继承来给对象增加功能来说,装饰器模式相比生成子类更为灵活. ...

  8. Python之旅Day5 列表生成式 生成器 迭代器 装饰器

    装饰器 器即函数,装饰即修饰,意指为其他函数添加新功能 装饰器定义:本质就是函数,功能是为其他函数添加新功能 装饰器涉及的知识点= 高阶函数+函数嵌套+闭包 在遵循下面两个原则的前提下为被装饰者新功能 ...

  9. 函数和常用模块【day05】:装饰器高潮(三)

    本节内容 1.概述 2.装饰器定义 3.装饰器定义 4.带参数的生成器 一.概述 我们之前介绍了大幅片的内容,感觉跟装饰器半毛钱关系都没有,其实不然,我们分别详细阐述了高阶函数和内置函数,下面我们就来 ...

随机推荐

  1. Codeforces 894.D Ralph And His Tour in Binary Country

    D. Ralph And His Tour in Binary Country time limit per test 2.5 seconds memory limit per test 512 me ...

  2. php使用 SImpleXMLElement 把 xml 和 数组 互转

    <?php $xml = <<<XML <xml> <ToUserName><![CDATA[toUser]]></ToUserNam ...

  3. python学习笔记(五) 200行实现2048小游戏

    用前文提到的基础知识,和网上流行的2048源码,用python实现该游戏. 先将用户操作和游戏逻辑绑定. WASD分别对应移动方向上.左.下.右 然后实现矩阵的转置和逆置,这样只要实现一个方向的移动, ...

  4. opencv函数制作的时钟模型

    http://www.cnblogs.com/sytu/p/4192652.html 在秒针模型的基础上添加了分针和时针,并且添加了暂停控件和设置时间的功能. #include"cv.h&q ...

  5. 「Linux」centos7更新python3.6后yum报错问题

    1. #vi /usr/bin/yum 因为我的旧版本是2.7,所以将#!/usr/bin/python改为#!/usr/bin/python2.7就可以了! 退出保存 2.可能还会报错 就修改/us ...

  6. OpenCV---环境安装和初次使用

    一:环境安装 pip3 install opencv-python #OpenCV模块,必须安装 pip3 install opencv-contrib-python #OpenCV扩展模块,选择安装 ...

  7. matlab的rem()和mod()函数

    matlab的rem()和mod()函数 rem(x,y):求整除x/y的余数 mod(x,y):求模 rem(x,y)=x-y.*fix(x./y);  (fix()向0取整) mod(x,y)=x ...

  8. dotnet core 实践——日志组件Serilog

     前几天把基于quartz.net的部分项目代码移植到了dotnet core ,但是没增加日志功能,原因是没找到合适的组件. 今天终于找到了Serilog: https://github.com/s ...

  9. Java并发编程学习路线

    一年前由于工作需要从微软技术栈入坑Java,并陆陆续续做了一个Java后台项目,目前在搞Scala+Java混合的后台开发,一直觉得并发编程是所有后台工程师的基本功,所以也学习了小一年Java的并发工 ...

  10. 2017ACM暑期多校联合训练 - Team 8 1006 HDU 6138 Fleet of the Eternal Throne (字符串处理 AC自动机)

    题目链接 Problem Description The Eternal Fleet was built many centuries ago before the time of Valkorion ...