Python装饰器重载内置的使用

 更新时间:2024年01月16日 09:46:14   作者:梯阅线条  
本文主要介绍了Python装饰器重载内置的使用,详细介绍如何创建装饰器,如何使用装饰器来重载内置函数,具有一定的参考价值,感兴趣的可以了解一下

python2.x的新式类和python3.x的全部类,内置操作表达式调用,不会触发__getattr__()和__getattribute__()。参考《python的getattr和getattribute拦截内置操作》。

NO内置操作表达式(隐式调用)对应方法(显式调用)
1索引操作[i]__getitem__
2加法(连接)操作+__coerce__ __add__
3括号调用()__call__
4打印print()__str__

__coerce__:表示强制类型转换,使用加法(或连接)操作+时,不同类型会触发类型转换或者报错。

内置操作调用方式:隐式调用,即调用表达式;显式调用,即调用方法名。

1.1 Py3内置操作表达式调用无法委托

描述

内置操作的表达式调用,无法在python3.0下委托,因为不会触发__getattr__()和__getattribute__()。

示例

>>> def traceCall(*args):#跟踪调用
    if trace:
        print('['+','.join(map(str,args))+']')

        
>>> def accessCtrl(forbid):
    def onDecorator(aCls):
        class onInstance:
            def __init__(self,*args,**kargs):
                self.__wrapped=aCls(*args,**kargs)
            def __getattr__(self,attr):
                traceCall('getattr',attr)
                if forbid(attr):
                    raise TypeError('禁止访问:'+attr)
                else:
                    return getattr(self.__wrapped,attr)
            def __setattr__(self,attr,value):
                traceCall('setattr',attr,value)
                # 压缩后的变量名为 _onInstance__wrapped
                if attr=='_onInstance__wrapped':
                    self.__dict__[attr]=value
                elif forbid(attr):
                    raise TypeError('禁止设置:'+attr)
                else:
                    setattr(self.__wrapped,attr,value)
        return onInstance
    return onDecorator

>>> def privateAttr(*privates):
    return accessCtrl(forbid=(lambda attr:attr in privates))

>>> @privateAttr('phone')
class Staff_Private:
    def __init__(self,name,phone):
        self.name=name
        self.phone=phone
    def __str__(self):
        return 'Staff_Private->手机号:'+str(self.phone)
    def __add__(self,num):
        self.phone+=num

Python2.x执行

Py2的__getattr__()拦截print()和+等内置操作的表达式调用,正确的委托给装饰对象。

#python2.x执行
>>> trace=True
>>> sp1=Staff_Private('梯阅线条',110)
[setattr,_onInstance__wrapped,Staff_Private->手机号:110]
#py2传统类 拦截内置操作表达式调用-隐式调用,print()
>>> print(sp1)
[getattr,__str__]
Staff_Private->手机号:110
#py2传统类 拦截内置操作表达式调用-隐式调用,+
>>> sp1+1
[getattr,__coerce__]
[getattr,__add__]
>>> print(sp1)
[getattr,__str__]
Staff_Private->手机号:111

Python3.x执行

Py3的__getattr__()不拦截print()和+等内置操作的表达式调用,无法委托给装饰对象。

#python3.x执行
>>> trace=True
>>> sp1=Staff_Private('梯阅线条',110)
[setattr,_onInstance__wrapped,Staff_Private->手机号:110]
#py3没有拦截内置操作 print
>>> print(sp1)
<__main__.accessCtrl.<locals>.onDecorator.<locals>.onInstance object at 0x0000019736C2F4F0>
#py3没有拦截内置操作 + 
>>> sp1+1
Traceback (most recent call last):
  File "<pyshell#11>", line 1, in <module>
    sp1+1
TypeError: unsupported operand type(s) for +: 'onInstance' and 'int'

1.2 装饰器重载内置操作

描述

Python3.x的装饰器重载内置操作运算符方法,来拦截装饰类对应的内置表达式调用。

比如,重载__str__()拦截print(),__add__()拦截+。

示例

>>> def traceCall(*args):#跟踪调用
    if trace:
        print('['+','.join(map(str,args))+']')
>>> def accessCtrl(forbid):
    def onDecorator(aCls):
        class onInstance:
            def __init__(self,*args,**kargs):
                self.__wrapped=aCls(*args,**kargs)
            def __getattr__(self,attr):
                traceCall('getattr',attr)
                if forbid(attr):
                    raise TypeError('禁止访问:'+attr)
                else:
                    return getattr(self.__wrapped,attr)
            def __setattr__(self,attr,value):
                traceCall('setattr',attr,value)
                # 压缩后的变量名为 _onInstance__wrapped
                if attr=='_onInstance__wrapped':
                    self.__dict__[attr]=value
                elif forbid(attr):
                    raise TypeError('禁止设置:'+attr)
                else:
                    setattr(self.__wrapped,attr,value)
            # print()触发__str__()
            def __str__(self):
                traceCall('onInstance,__str__')
                # str()触发__str__()
                return str(self.__wrapped)
            # + 触发 __add__()    
            def __add__(self,other):
                traceCall('onInstance,__add__',other)
                return self.__wrapped+other
        return onInstance
    return onDecorator

>>> def privateAttr(*privates):
    return accessCtrl(forbid=(lambda attr:attr in privates))

>>> @privateAttr('phone')
class Staff_Private:
    def __init__(self,name,phone):
        self.name=name
        self.phone=phone
    def __str__(self):
        traceCall('Staff_Private,__str__')
        return 'Staff_Private->手机号:'+str(self.phone)
    def __add__(self,num):
        traceCall('Staff_Private,__add__',num)
        self.phone+=num

        
>>> trace=True
>>> sp1=Staff_Private('梯阅线条',110)
[Staff_Private,__str__]#traceCall调用print()触发__str__()
[setattr,_onInstance__wrapped,Staff_Private->手机号:110]
# print()触发__str__()
>>> print(sp1)
[onInstance,__str__]
[Staff_Private,__str__]
Staff_Private->手机号:110
# + 触发 __add__()
>>> sp1+1
[onInstance,__add__,1]
[Staff_Private,__add__,1]
>>> print(sp1)
[onInstance,__str__]
[Staff_Private,__str__]
Staff_Private->手机号:111

到此这篇关于Python装饰器重载内置的使用的文章就介绍到这了,更多相关Python装饰器重载内置内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • python中base64编码简介

    python中base64编码简介

    Base64是一种任意二进制到文本字符串的编码方法,常用于在URL、Cookie、网页中传输少量二进制数据,Base64是一种用64个字符来表示任意二进制数据的方法,这篇文章主要介绍了python中base64编码,需要的朋友可以参考下
    2022-12-12
  • Python实现数值积分方式

    Python实现数值积分方式

    今天小编就为大家分享一篇Python实现数值积分方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-11-11
  • windows系统中Python多版本与jupyter notebook使用虚拟环境的过程

    windows系统中Python多版本与jupyter notebook使用虚拟环境的过程

    这篇文章主要介绍了windows中Python多版本与jupyter notebook中使用虚拟环境的安装过程,本文分步骤给大家介绍的非常详细,具有一定的参考借鉴价值 ,需要的朋友可以参考下
    2019-05-05
  • 详解Python中with语句的用法

    详解Python中with语句的用法

    这篇文章主要介绍了Python中with语句的用法,with语句的使用是Python学习过程当中的基础知识,本文来自于IBM官方技术文档,需要的朋友可以参考下
    2015-04-04
  • 解决List.append() 在 Python 中不起作用的问题

    解决List.append() 在 Python 中不起作用的问题

    在 Python 中,我们通常使用 List.append() 方法向列表末尾添加元素,然而,在某些情况下,你可能会遇到 List.append() 方法不起作用的问题,本文将详细讨论这个问题并提供解决方法,需要的朋友可以参考下
    2023-06-06
  • Python设计模式之外观模式实例详解

    Python设计模式之外观模式实例详解

    这篇文章主要介绍了Python设计模式之外观模式,结合实例形式详细分析了外观模式的概念、原理、用法及相关操作注意事项,需要的朋友可以参考下
    2019-01-01
  • python使用锁访问共享变量实例解析

    python使用锁访问共享变量实例解析

    这篇文章主要介绍了python使用锁访问共享变量实例解析,分享了相关代码示例,小编觉得还是挺不错的,具有一定借鉴价值,需要的朋友可以参考下
    2018-02-02
  • Python实现对桌面进行实时捕捉画面的方法详解

    Python实现对桌面进行实时捕捉画面的方法详解

    最近在研究目标检测方面的小东西,需要到对桌面进行实时捕捉画面。所以本文来用Python实现简单的对桌面进行实时捕捉画面,感兴趣的可以了解一下
    2023-01-01
  • Python制作一个随机抽奖小工具的实现

    Python制作一个随机抽奖小工具的实现

    最近在工作中面向社群玩家组织了一场活动,需要进行随机抽奖,就做了一个简单的随机抽奖小工具。具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-07-07
  • python中get和post有什么区别

    python中get和post有什么区别

    在本篇内容里小编给大家分享的是关于python中get和post有什么区别的相关内容,需要的朋友们参考下吧。
    2020-06-06

最新评论