首页 > 编程语言 >Python装饰器

Python装饰器

时间:2023-04-21 09:22:21浏览次数:42  
标签:return Python ret func time print 装饰 def

装饰器的作用 —— 不想修改函数的调用方式,但是还想在原来的函数前后添加功能
原则 :开放封闭原则
  开放 : 对扩展是开放的
  封闭 : 对修改是封闭的

装饰器的本质 : 闭包函数

 

装饰器用法  

装饰不带参数的函数

import time

def timmer(func):
    def inner():
        start_timev= time.time()
        data = func()
        end_time = time.time()
        print(end_time-start_timev)
        return data
    return  inner

@timmer
def func():
    time.sleep(0.5)
    print("不带参数的函数,等会我要装饰下自己")
    return '我是函数'

ret = func()
print(ret)
View Code

 

装饰带参数的函数

def timmer(func):
    def inner(*args, **kwargs):
        start_timev= time.time()
        data = func(*args, **kwargs)  ##参数灵活设置,不固定
        end_time = time.time()
        print(end_time-start_timev)
        return data
    return  inner

@timmer
def func(a, b):
    time.time(0.5)
    print(f'我是参数{a}, 我是另一个参数{b}')
    return a+b

ret = func('hello', 'girls')
print(ret)
View Code

 

装饰器带参数,可根据场景动态实现函数是否需要装饰

#带参数的装饰器
FLAG = False   ###通过配置表设置函数是否需要被装饰
def timmer(flag):
    def wrapper(func):
        def inner(*args, **kwargs):
            if flag:
                start_timev = time.time()
                data = func(*args, **kwargs)
                end_time = time.time()
                print('函数执行时间差', end_time - start_timev)
                return data
            else:
                data = func(*args, **kwargs)
                return data
        return inner
    return wrapper

@timmer(FLAG)
def func1(a, b):
    time.sleep(0.5)
    print(f'我是参数{a}, 我是另一个参数{b}')
    return a+b

@timmer(FLAG)
def func2(l):
    time.sleep(0.5)
    print(type(l), f'我是大列表')
    return '我是列表我是老大'

func1('hello', 'girls')
func2([1,3,5,'a', 's', 8, 'k'])
View Code

 

多个装饰器装饰同一个函数

def wrapper1(func):
    def inner1():
        print('wrapper1 ,before func')
        ret = func()
        print('wrapper1 ,after func')
        return ret
    return inner1

def wrapper2(func):
    def inner2():
        print('wrapper2 ,before func')
        ret = func()
        print('wrapper2 ,after func')
        return ret
    return inner2

def wrapper3(func):
    def inner3():
        print('wrapper3 ,before func')
        ret = func()
        print('wrapper3 ,after func')
        return ret
    return inner3

@wrapper3
@wrapper2
@wrapper1
def f():
    print('in f')
    return '哈哈哈'

print(f())
View Code

 

functools.wraps:保留原函数信息

from functools import wraps
def wrapper(func):  #func = holiday
    @wraps(func)
    def inner(*args,**kwargs):
        print('在被装饰的函数执行之前做的事')
        ret = func(*args,**kwargs)
        print('在被装饰的函数执行之后做的事')
        return ret
    return inner

@wrapper   #holiday = wrapper(holiday)
def holiday(day):
    '''这是一个放假通知'''
    print('全体放假%s天'%day)
    return '好开心'

print(holiday.__name__)
print(holiday.__doc__)  ##函数注释
ret = holiday(3)   #inner
print(ret)
View Code

 

demo1:用户登录才能加车或删除商品

FLAG = False

def login(func):
    def inner(*args, **kwargs):
        global FLAG
        if FLAG:
            return func(*args, **kwargs)
        username = input('请输入用户名: ')
        pwd = input('请输入密码: ')

        if username=='myname' and pwd == '123':
            FLAG = True
            return func(*args, **kwargs)
        else:
            print('login fail')
    return inner

@login
def shoplist_add():
    print('增加一件物品')

@login
def shoplist_del():
    print('删除一件物品')

shoplist_add()
shoplist_del()
View Code

 

demo2:为调用函数加上记录调用功能,要求每次调用函数都将被调用的函数名称写入文件

def log(func):
    def inner(*args,**kwargs):
        with open('log','a',encoding='utf-8') as f:
            f.write(func.__name__+'\n')
        ret = func(*args,**kwargs)
        return ret
    return inner

@log
def shoplist_add():
    print('增加一件物品')

@log
def shoplist_del():
    print('删除一件物品')
View Code

 

标签:return,Python,ret,func,time,print,装饰,def
From: https://www.cnblogs.com/margret/p/17337744.html

相关文章

  • python通过psutil获取服务器cpu,内存,磁盘使用率
    psutil是一个跨平台的Python库,它允许你获取有关系统进程和系统资源使用情况的信息。它支持Windows、Linux、OSX、FreeBSD、OpenBSD和NetBSD等操作系统,并提供了一些非常有用的功能,如:获取系统CPU使用率、内存使用率、磁盘使用率等信息。获取进程列表、进程状态、进程CPU使用率、......
  • java调用python脚本,用到tensorflow、keras等第三方库
    https://blog.csdn.net/jstlovely/article/details/121247764?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522168110434116800227452800%2522%252C%2522scm%2522%253A%252220140713.130102334.pc%255Fall.%2522%257D&request_id=168110434116800227452800&......
  • python_面向对象
    魔法函数魔术访达的一些说明23-24魔术方法并不是来自继承,而是python自带的这些方法,跟类没有关系__getitem__方法因为实现了__getitem__(),所以可以直接遍历这个student对象__getitem__方法的参数item就是相当于数组的下标13行对象实例化对象进行切片操作......
  • 【进阶15】【自学笔记】Python运行cmd命令的几种方式
    一、pathlib的简单介绍pathlib是Python3.4及更高版本中内置的标准库,提供了一种面向对象的方式来处理文件系统路径。它为不同操作系统提供了合适的路径语义,并支持常见的文件和目录操作,比如判断路径是否存在、获取路径的各个部分、创建/删除目录等操作。二、基本操作1、获取......
  • Python 判断服务端口是否被占用脚本
    背景自动注册服务时判断端口是否被占用代码详情importsocketdefcheck_port(address,port):#创建套接字对象sock=socket.socket(socket.AF_INET,socket.SOCK_STREAM)#检查端口是否占用result=sock.connect_ex((address,port))#关闭套接字......
  • 【进阶14】【自学笔记】Python运行cmd命令的几种方式
    1、使用os.system()函数importos#运行cmd命令os.system('dir')2、使用subprocess模块importsubprocess#运行cmd命令subprocess.run(['dir'],shell=True)3、使用os.popen()函数importos#运行cmd命令result=os.popen('dir')print(result.read......
  • 对比Python中的列表、元组、字典、集合、字符串等之间异同
    1.数据类型列表、元组、字典、集合、字符串均属于python3的标准数据类型。字符串和元组属于不可变数据,即创建后不可修改。列表、字典、集合属于可变数据,即创建后可以修改元素。2.创建有元素的对象3.创建没有元素的对象列表使用eval()或list()或中括号[]进行创建,元素之间使用逗号分......
  • Linux 编译安装 Python3
    本文档适用Python3.9及以上版本。1.提前安装依赖yuminstall-ygccncurses-develgdbm-develxz-develsqlite-develtk-develuuid-develreadline-develbzip2-devellibffi-developenssl11openssl11-devel2.设置编译FLAGAWSAMI:exportCFLAGS=$(pkg-config--cflagsli......
  • #yyds干货盘点#python之 Lambda 表达式
    lambda 关键字用于创建小巧的匿名函数。lambda a, b: a+b 函数返回两个参数的和。Lambda函数可用于任何需要函数对象的地方。在语法上,匿名函数只能是单个表达式。在语义上,它只是常规函数定义的语法糖。与嵌套函数定义一样,lambda函数可以引用包含作用域中的变量:>>>defmake_......
  • [oeasy]python0135_变量名与下划线_dunder_声明与赋值
    变量定义回忆上次内容变量就是能变的量上次研究了变量标识符的规则第一个字符应该是字母或下划线合法的标识符可以包括大小写字母数字下划线  还研究了字符串(str)的函数isidentifier查询字符串是否为合法标识符 ......