python装饰器、描述符模拟源码分析

概要

本人python理论知识远达不到传授级别,写文章主要目的是自我总结,并不能照顾所有人,请见谅,文章结尾贴有相关链接可以作为补充

全文分为三个部分装饰器理论知识、装饰器应用、装饰器延申

  • 装饰理基础:无参装饰器、有参装饰器、functiontools、装饰器链
  • 装饰器进阶:property、staticmethod、classmethod源码分析(python代码实现)

装饰器基础

  • 无参装饰器
'''
假定有一个需求是:打印程序函数运行顺序
此案例打印的结果为:
    foo1 function is starting
    foo2 function is starting
'''
from functools import wraps


def NoParamDec(func):
    #函数在被装饰器装时后,其函数属性也会改变,wraps作用就是保证被装饰函数属性不变
    @wraps(func)
    def warpper(*args, **kwargs):
        print('{} function is starting'.format(func.__name__))
        return func(*args, **kwargs)
    
    return warpper


#python黑魔法省略了NoParamDec=NoParamDec(foo1)
@NoParamDec
def foo1():
    foo2()

@NoParamDec
def foo2():
    pass


if __name__ == "__main__":

    foo1()

  • 有参装饰器
'''
假定有一个需求是:检查函数参数的类型,只允许匹配正确的函数通过程序
此案例打印结果为:
('a', 'b', 'c')
-----------------------分割线------------------------
ERROS!!!!b must be <class 'str'> 
ERROS!!!!c must be <class 'str'> 
('a', 2, ['b', 'd'])

    
'''
from functools import wraps
from  inspect import signature


def typeAssert(*args, **kwargs):
    deco_args = args
    deco_kwargs = kwargs
    
    def factor(func):
        #python标准模块类,可以用来检查函数参数类型,只允许特定类型通过
        sig = signature(func)
        #将函数形式参数和规定类型进行绑定
        check_bind_args = sig.bind_partial(*deco_args, **deco_kwargs).arguments
        
        @wraps(func)
        def wrapper(*args, **kwargs):
            #将实际参数值和形式参数进行绑定
            wrapper_bind_args = sig.bind(*args, **kwargs).arguments.items()
            for name, obj in wrapper_bind_args:
                #遍历判断是否实际参数值是规定参数的实例
                if not isinstance(obj, check_bind_args[name]):
                    try:
                        raise TypeError('ERROS!!!!{arg} must be {obj} '.format(**{'arg': name, 'obj': check_bind_args[name]}))
                    except Exception as e:
                        print(e)
            return func(*args, **kwargs)
        
        return wrapper
    
    return factor


@typeAssert(str, str, str)
def inspect_type(a, b, c):
    return (a, b, c)


if __name__ == "__main__":
    print(inspect_type('a', 'b', 'c'))
    print('{:-^50}'.format('分割线'))
    print(inspect_type('a', 2, ['b', 'd']))


  • 装饰器链
'''
假定有一个需求是:
输入类似代码:
@makebold
@makeitalic
def say():
   return "Hello"

输出:
<b><i>Hello</i></b>
'''
from functools import wraps


def html_deco(tag):
    def decorator(fn):
        @wraps(fn)
        def wrapped(*args, **kwargs):
            return '<{tag}>{fn_result}<{tag}>'.format(**{'tag': tag, 'fn_result': fn(*args, **kwargs)})
        
        return wrapped
    
    return decorator


@html_deco('b')
@html_deco('i')
def greet(whom=''):
    # 等价于 geet=html_deco('b')(html_deco('i)(geet))
    return 'Hello' + (' ' + whom) if whom else ''


if __name__ == "__main__":
    print(greet('world'))  # -> <b><i>Hello world</i></b>


装饰器进阶

  • property 原理
    通常,描述符是具有“绑定行为”的对象属性,其属性访问已经被描述符协议中的方法覆盖。这些方法是get()、set()和delete()。如果一个对象定义这些方法中的任何一个,它被称为一个描述符。如果对象定义get()和set(),则它被认为是数据描述符。仅定义get()的描述器称为非数据描述符(它们通常用于方法,但是其他用途也是可能的)。

属性查找优先级为:

  • 类属性
  • 数据描述符
  • 实例属性
  • 非数据描述符
  • 默认为getattr()
class Property(object):
    '''
    内部property是用c实现的,这里用python模拟实现property功能
    代码参考官方doc文档
    '''

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

    def __get__(self, obj, objtype=None):
        if obj is None:
            return self
        if self.fget is None:
            raise (AttributeError, "unreadable attribute")
        print('self={},obj={},objtype={}'.format(self,obj,objtype))
        return self.fget(obj)

    def __set__(self, obj, value):
        if self.fset is None:
            raise (AttributeError, "can't set attribute")
        self.fset(obj, value)

    def __delete__(self, obj):
        if self.fdel is None:
            raise (AttributeError, "can't delete attribute")
        self.fdel(obj)

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

    def setter(self, fset):
        return type(self)(self.fget, fset, self.fdel, self.__doc__)

    def deleter(self, fdel):
        return type(self)(self.fget, self.fset, fdel, self.__doc__)


class Student( object ):
    @Property
    def score( self ):
        return self._score
    @score.setter
    def score( self, val ):
        if not isinstance( val, int ):
            raise ValueError( 'score must be an integer!' )
        if val > 100 or val < 0:
            raise ValueError( 'score must between 0 ~ 100!' )
        self._score = val


if __name__ == "__main__":
    s = Student()
    s.score = 60   
    s.score         
  • staticmethod 原理
    @staticmethod means: when this method is called, we don't pass an instance of the class to it (as we normally do with methods). This means you can put a function inside a class but you can't access the instance of that class (this is useful when your method does not use the instance).
class StaticMethod(object):
    "python代码实现staticmethod原理"
    
    def __init__(self, f):
        self.f = f
    
    def __get__(self, obj, objtype=None):
        return self.f


class E(object):
    #StaticMethod=StaticMethod(f)
    @StaticMethod
    def f( x):
        return x

if __name__ == "__main__":
    print(E.f('staticMethod Test'))

  • classmethod
    @staticmethod means: when this method is called, we don't pass an instance of the class to it (as we normally do with methods). This means you can put a function inside a class but you can't access the instance of that class (this is useful when your method does not use the instance).
class ClassMethod(object):
    "python代码实现classmethod原理"
    
    def __init__(self, f):
        self.f = f
    
    def __get__(self, obj, klass=None):
        if klass is None:
            klass = type(obj)
        
        def newfunc(*args):
            return self.f(klass, *args)
        
        return newfunc
    
class E(object):
    #ClassMethod=ClassMethod(f)
    @ClassMethod
    def f(cls,x):
        return x
    
if __name__ == "__main__":
    print(E().f('classMethod Test'))

参考资料

1, statckoverflow: how to make a chain of decorators

2, python doc:how to descriptor

3,知乎:如何理解装饰器

4, difference-between-staticmethod-and-classmethod-in-python

5,meaning-of-classmethod-and-staticmethod-for-beginner

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,826评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,968评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,234评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,562评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,611评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,482评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,271评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,166评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,608评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,814评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,926评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,644评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,249评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,866评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,991评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,063评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,871评论 2 354