#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Created on Tue May 5 21:40:49 2020 @author: root 装饰器语法糖
装饰器的使用方法很固定: 1.先定义一个装饰函数(帽子)(也可以用类、偏函数实现)
2.再定义你的业务函数、或者类(人)
3.最后把这顶帽子带在这个人头上 案例:https://www.jb51.net/article/168276.htm
""" #2日志打印器
# 首先是日志打印器。
# 它要实现的功能是
# 在函数执行前,先打印一行日志告知一下主人,我要执行函数了。
# 在函数执行完,也不能拍拍屁股就走人了,咱可是有礼貌的代码,再打印一行日志告知下主人,我执行完啦。
def logger(func):
def wrapper(*args,**kwargs):
print ('start--------------')
func(*args,**kwargs)
print ('end----------------')
return wrapper @logger
def add(x,y):
print('{} + {} = {}'.format(x, y, x+y)) add(20,90) #3 入门用法:时间计时器
import time
def timer(func):
def wrapper(*args,**kw):
t1=time.time()
func(*args,**kw)
t2=time.time()
cost_time=t2-t1
print ("花费时间:{}秒".format(cost_time))
return wrapper @timer
def want_sleep(sleep_time):
time.sleep(sleep_time)


want_sleep(10)    

# 3进阶用法:带参数的函数装饰器
def say_hello(contry):
def wrapper(func):
def deco(*args,**kwargs):
if contry=='china':
print ("ni hao")
elif contry=='american':
print ('hello')
else:
return
func(*args,**kwargs)
return deco
return wrapper @say_hello('american')
def american():
print ('----------------china') @say_hello('china')
def chinese():
print ('----------------zhongguo') american()
print ('---------------')
chinese() # 高阶用法:不带参数的类装饰器
"""
以上都是基于函数实现的装饰器,在阅读别人代码时,还可以时常发现还有基于类实现的装饰器。
基于类装饰器的实现,必须实现 __call__ 和 __init__两个内置函数。 __init__ :接收被装饰函数
__call__ :实现装饰逻辑
"""
class logger(object):
def __init__(self,func):
self.func=func def __call__(self,*args,**kwargs):
print("[INFO]: the function {func}() is running..."\
.format(func=self.func.__name__))
return self.func(*args,**kwargs) @logger
def say(something):
print ("say {}!".format(something))
say('hello') # 高阶用法:带参数的类装饰器
# 上面不带参数的例子,你发现没有,只能打印INFO级别的日志,正常情况下,我们还需要打印DEBUG WARNING等级别的日志。 这就需要给类装饰器传入参数,给这个函数指定级别了。
# 带参数和不带参数的类装饰器有很大的不同。 # __init__ :不再接收被装饰函数,而是接收传入参数。
# __call__ :接收被装饰函数,实现装饰逻辑 class logger2(object):
def __init__(self,level='INFO'):
self.level=level def __call__(self,func):
def wrapper(*args,**kwargs):
print("[{level}]: the function {func}() is running..."\
.format(level=self.level, func=func.__name__))
func(*args, **kwargs)
return wrapper @logger2(level='warning')
def say(something):
print ("say {}!".format(something))
say('hello') # 内置装饰器:property
# 以上,我们介绍的都是自定义的装饰器。
# 其实Python语言本身也有一些装饰器。比如property这个内建装饰器,我们再熟悉不过了。
# 它通常存在于类中,可以将一个函数定义成一个属性,属性的值就是该函数return的内容。 class Student(object):
def __init__(self, name):
self.name = name
self.name = None @property
def age(self):
return self._age @age.setter
def age(self, value):
if not isinstance(value, int):
raise ValueError('输入不合法:年龄必须为数值!')
if not 0 < value < 100:
raise ValueError('输入不合法:年龄范围必须0-100')
self._age=value @age.deleter
def age(self):
del self._age XiaoMing = Student("小明") # 设置属性
XiaoMing.age = 25 print (XiaoMing.name)
print (XiaoMing.age) # example2:
class Student:
def __init__(self,name):
self.name=name @property
def math(self):
return self._math @math.setter
def math(self,value):
if 0<=value<=100:
self._math=value
else:
raise ValueError("Valid value must be in [0, 100]")

实现propetry 装饰器


class TestProperty(object):
    def __init__(self, fget=None, fset=None, fdel=None, doc=None):
        self.fset = fset
        self.fget = fget
        self.fdel = fdel
        self.__doc__ = doc

    def __get__(self, obj, objtype=None):
        print('---in --get')
        if obj is None:
            return self
        if self.fget is None:
            raise AttributeError
        return self.fget(obj)

    def __set__(self,obj,value):
        print ('----in --set')
        if self.fset is None:
            raise AttributeError
        self.fset(obj,value)

    def __delete__(self,obj):
        print ('---in ---delete')
        if self.fdel is None:
           raise  AttributeError
        self.fdel(obj)

    def getter(self,fget):
        print ('in getter')
        return type(self)(fget,self.fset,self.fdel,self.__doc__)

    def setter(self,fset):
        print ('in setter')
        return type(self)(fset,self.fget,self.fdel,self.__doc__)
    
    def deleter(self,fdel):
        print ('in fdel')
        return type(self)(fdel,self.fget,self.fset,self.__doc__)

测试一下:

class Student:
def __init__(self, name):
self.name = name # 其实只有这里改变
@TestProperty
def math(self):
return self._math @math.setter
def math(self, value):
if 0 <= value <= 100:
self._math = value
else:
raise ValueError("Valid value must be in [0, 100]")

python 装饰函数2的更多相关文章

  1. Python装饰函数

    from time import ctime, sleep def tsfunc(func): def wrappedFunc(): print('[%s] %s() classed' % (ctim ...

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

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

  3. python基础—函数装饰器

    python基础-函数装饰器 1.什么是装饰器 装饰器本质上是一个python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外功能. 装饰器的返回值是也是一个函数对象. 装饰器经常用于有切 ...

  4. Python装饰器、内置函数之金兰契友

    装饰器:装饰器的实质就是一个闭包,而闭包又是嵌套函数的一种.所以也可以理解装饰器是一种特殊的函数.因为程序一般都遵守开放封闭原则,软件在设计初期不可能把所有情况都想到,所以一般软件都支持功能上的扩展, ...

  5. python之函数对象、函数嵌套、名称空间与作用域、装饰器

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

  6. python -- 返回函数、匿名函数、装饰器

    返回函数 高阶函数的参数可以是函数,那么其返回值也可以是函数. 闭包 对于高阶函数,内部函数可以引用外部函数的参数和局部变量.当调用外部函数返回内部函数时,相关参数和变量都保存在返回的函数(即内部函数 ...

  7. python装饰器1:函数装饰器详解

    装饰器1:函数装饰器 装饰器2:类装饰器 装饰器3:进阶 先混个眼熟 谁可以作为装饰器(可以将谁编写成装饰器): 函数 方法 实现了__call__的可调用类 装饰器可以去装饰谁(谁可以被装饰): 函 ...

  8. Python之函数对象、函数嵌套、名称空间与作用域、闭包函数、装饰器

    目录 一 函数对象 二 函数的嵌套 三 名称空间和作用域 四 闭合函数 五 装饰器 一.函数对象 1.函数是第一类对象 #第一类对象的特性:# 可以被引用 # 可以当做参数传递 # 返回值是函数 # ...

  9. Python之函数(自定义函数,内置函数,装饰器,迭代器,生成器)

    Python之函数(自定义函数,内置函数,装饰器,迭代器,生成器) 1.初始函数 2.函数嵌套及作用域 3.装饰器 4.迭代器和生成器 6.内置函数 7.递归函数 8.匿名函数

随机推荐

  1. Java如何使用while和for嵌套循环控制输出数据,使数据奇偶行不同

    /* 题目1 使用Eclipse编写控制台应用程, 使用while循环在控制台打印10行10列的如下图形 □ □ □ □ □ □ □ □ □ □ ■ ■ ■ ■ ■ ■ ■ ■ ■ ■ □ □ □ □ ...

  2. 十七、.net core(.NET 6)搭建基于Quartz组件的定时调度任务

     搭建基于Quartz组件的定时调度任务 先在package包项目下,添加Quartz定时器组件: 新建类库项目Wsk.Core.QuartzNet,并且引用包类库项目.然后新建一个中间调度类,叫Qu ...

  3. NOIP模拟测试13「矩阵游戏&#183;跳房子&#183;优美序列」

    矩阵游戏 考试时思路一度和正解一样,考试到最后还是打了80分思路,结果80分打炸了只得了40分暴力分 题解 算出来第一列的总值,每次通过加每两列之间的差值得出下一列的总值 算第一列我们只需要让当前点* ...

  4. FTP下载文件时拒绝登陆申请怎么办?

    1.有时候用网页登陆FTP无法下载文件,如下图 2.这时候就需要用文件夹登陆FTP,(打开我的电脑,然后输入ftp://10.2.41.31.如下图 然后就可以下载文件了

  5. 最佳实践丨使用Rancher轻松管理上万资源不是梦!

    前 言 Rancher 作为一个开源的企业级 Kubernetes 集群管理平台.你可以导入现有集群,如 ACK.TKE.EKS.GKE,或者使用 RKE.RKE2.K3s 自定义部署集群. 作为业界 ...

  6. App免责声明

    一切移动客户端用户在下载并浏览xxxAPP软件时均被视为已经仔细阅读本条款并完全同意.凡以任何方式使用本APP,或直接.间接使用本APP资料者,均被视为自愿接受本网页相关声明和用户服务协议的约束. x ...

  7. 20 批量ping主机

    #!/bin/bash IP_LIST="`cat /root/ip.txt`" ip_dir="/root/ip_dir" [ ! -d ${ip_dir} ...

  8. layui 监听checkbox 、radio、switch选中改变事件

    form.on('checkbox(filter)', function(data){ console.log(data.elem); //得到checkbox原始DOM对象 console.log( ...

  9. CRM系统为什么达不到预期效果?

    随着信息技术的发展,企业对于信息化转型的需求越发强烈,而CRM客户关系管理系统成为了企业信息化转型的首选.尽管CRM系统对于企业有着很重要的作用,但有不少企业在选型和实施时遇到了问题,导致CRM系统没 ...

  10. webpack(5)webpack处理css文件

    css文件处理-准备工作 (以下项目配置都是基于上一篇webpack(4)的基础上) 在项目开发中,我们必然需要添加很多的样式,而样式我们往往写到一个单独的文件中. 这里我们就在src目录中创建一个n ...