Python类装饰器跟踪委托对象的属性访问

1 Python类装饰器跟踪委托对象的属性访问

类装饰器通过添加逻辑层,管理实例接口的访问。

通过委托的方式类跟踪对象属性的访问。

通过拦截未定义属性的访问,并且转向委托对象的属性访问,来实现委托。

1.1 非装饰器委托

描述

python通过__getattr__()拦截未定义属性,再使用getattr()访问委托对象的属性,实现委托对象的访问。

(1) 定义一个类,构造函数内将委托对象赋值给实例属性;

(2) 重载__getattr__()方法,拦截未定义属性的点号运算,使用getattr()转到委托对象的属性访问;已定义属性的点号运算不会被拦截;

示例

>>> class MyWrapper:def __init__(self,obj):# 将委托对象赋值给实例属性self.obj=objdef __getattr__(self,attr):# 拦截未定义属性的点号运算print('atrr:',attr)# 通过委托对象获取属性return getattr(self.obj,attr)>>> mwl=MyWrapper(list('梯阅线条'))
>>> mwl.append(9555)
atrr: append
# 已定义属性不触发 __getattr__() 方法
>>> mwl.obj
['梯', '阅', '线', '条', 9555]
>>> mwd=MyWrapper({'a':1,'b':2})
>>> tuple(mwd.keys())
atrr: keys
('a', 'b')

1.2 类装饰器委托

描述

python可以通过函数编写的类装饰器来实现委托,进行跟踪委托对象的属性访问。

(1) 定义一个函数为类装饰器,参数接收一个类,内部定义一个类,并且返回这个类;

(2) 类装饰器需要返回可调用对象,因为装饰类创建实例对象时,会调用()小括号;

(3) 将装饰类的实例对象赋值给装饰器的实例属性;

(4) 装饰器重载__getattr__()拦截未定义属性的点号运算,即拦截装饰类实例属性;

(5) 将未定义属性通过getattr()转到装饰器类实例的属性获取实现委托;

示例

>>> def DecoWrapper(cls):#类装饰器接收一个类参数class MyWrapper:def __init__(self,*args,**kargs):# 将装饰类的实例对象赋值给装饰器的实例属性self.cls=cls(*args,**kargs)self.gets=0def __getattr__(self,attr):#拦截未定义属性的点号运算print('atrr:',attr)self.gets+=1# 将未定义属性通过装饰类的实例对象进行获取实现委托return getattr(self.cls,attr)return MyWrapper#类装饰器返回可调用对象>>> @DecoWrapper
class TestDW:def doubles(self):print('梯阅线条'*2)>>> @DecoWrapper
class Clerk:def __init__(self,name,rate,days):self.name=nameself.rate=rateself.days=daysdef pay(self):return self.rate*self.days# 通过类名()的方式创建类的实例对象
# 即()小括号运算创建实例对象,触发__init__()构造方法 
>>> tdw1=TestDW()
>>> tdw1.doubles()
atrr: doubles
梯阅线条梯阅线条
>>> tdw1.gets
1
>>> c1=Clerk('梯阅线条',2000,22)
>>> c2=Clerk(9555,3000,23)
>>> c1.name,c1.pay()
atrr: name
atrr: pay
('梯阅线条', 44000)
>>> c2.name,c2.pay()
atrr: name
atrr: pay
(9555, 69000)
>>> c1.gets,c2.gets
(2, 2)
>>> c1.name
atrr: name
'梯阅线条'
>>> c1.gets,c2.gets
(3, 2)

1.3 类装饰器-类无法保存多个实例

描述

类装饰器用类实现的时候,同一个装饰类无法保存多个装饰类的实例对象。

因为用类实现类装饰器时,重载__call__()方法,返回装饰器实例对象,创建多个相同装饰类的实例对象时,相当于执行同一个装饰器实例对象,并且只保存了最后一个实例的属性。

示例

>>> class DecoWrapper:def __init__(self,cls):# 将装饰类赋值给装饰器的实例属性self.cls=clsself.gets=0def __call__(self,*args):# 装饰类小括号()运算触发 __call__() 方法# 装饰类实例对象赋值给装饰器类实例属性self.wrapped=self.cls(*args)# 需返回装饰器实例对象,# 否则后续装饰类实例对象无法进行属性拦截运算return selfdef __getattr__(self,attr):#拦截未定义属性的点号运算print('atrr:',attr)self.gets+=1# 将未定义属性通过装饰类的实例对象进行获取实现委托return getattr(self.wrapped,attr)# @DecoWrapper 将 DecoWrapper 类的实例对象赋值给 Clerk
# 触发 DecoWrapper 的 __init__()方法
>>> @DecoWrapper
class Clerk:def __init__(self,name,rate,days):self.name=nameself.rate=rateself.days=daysdef pay(self):return self.rate*self.days# Clerk()小括号运算触发 __call__()方法 
>>> c1=Clerk('梯阅线条',2000,22)
>>> id(c1)
68722448
# 属性点号运算触发 __getattr__() 方法
>>> c1.name,c1.pay()
atrr: name
atrr: pay
('梯阅线条', 44000)
# Clerk没变化,相当于再次调用 __call__() 方法
# 修改了 self.wrapped 的属性值
>>> c2=Clerk(9555,3000,23)
>>> id(c2)
68722448
# c1 和 c2 指向同一个对象
# 都指向修改后的 self.wrapped 的属性值
>>> id(c1)==id(c2)
True
>>> c2.name,c2.pay()
atrr: name
atrr: pay
(9555, 69000)
# c1的值被c2修改
>>> c1.name,c1.pay()
atrr: name
atrr: pay
(9555, 69000)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/616993.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

结构体成员 分数比较大小

题目&#xff1a; 代码&#xff1a; #include <bits/stdc.h> #include<cstring>using namespace std;struct Num{double fenzi;double fenmu;char fenhao;};bool cmp(Num r1,Num r2){return r1.fenzi/r1.fenmu<r2.fenzi/r2.fenmu;}int main(){int n;Num num[n…

Android-多线程

线程是进程中可独立执行的最小单位&#xff0c;也是 CPU 资源&#xff08;时间片&#xff09;分配的基本单位&#xff0c;同一个进程中的线程可以共享进程中的资源&#xff0c;如内存空间和文件句柄。线程有一些基本的属性&#xff0c;如id、name、以及priority。 id&#xff1…

管理沟通能力测试

管理沟通能力测试用于测试管理者的潜能在交流和沟通方面的能力&#xff0c;作为管理者的能力测评是一个综合面&#xff0c;而沟通能力则是尤为重要&#xff0c;团队的凝聚力、创造力都跟管理者的沟通能力有着直接的关系。了解和提高管理沟通能力需要在实践中不断学习和总结。 …

C++(20):vector通过erase,erase_if删除符合条件的元素

C++20前,vector可以通过成员函数erase删除迭代器指定的元素,并返回被删除的下一个元素: iterator erase( iterator pos ); iterator erase( iterator first, iterator last ); 1.删除单个元素 #include <vector> #include <iostream> #include <algorithm&…

php 字符串常用函数

目录 1.一些常用函数 2.代码示例 1.一些常用函数 函数名描述trim()删除字符串两端空行或其它预定义符rtrim()删除字符串右边空行或其它预定义符ltrim()删除字符串左边空行或其它预定义符dirname()返回路径中的目录部分str_split()把字符串分割到数组里explode()使用一个字符串…

写在学习webkit过程的前面

webkit起源于KHTML&#xff0c;是KDE开源项目的KHTML和KJS引擎的一部分。在它的诞生和发展过程中&#xff0c;由两家著名的公司参与开发过程中&#xff0c;造成两次裂变。诞生两个内核webkit和blink&#xff0c;并发展和产生了两个主流的浏览器&#xff0c;分别为safari和chrom…

c++学习笔记-STL案例-机房预约系统2-创建身份类

前言 衔接上一篇“c学习笔记-STL案例-机房预约系统1-准备工作”&#xff0c;本文主要包括&#xff1a;创建身份类&#xff0c;建立了整个系统的框架&#xff0c;Identity基类&#xff0c;派生类&#xff1a;Sudent、Teacher、Manager&#xff0c;基类无实现源文件&#xff0c;…

鸿蒙HarmonyOS兼容JS的类Web开发-开发指导

鸿蒙HarmonyOS兼容JS的类Web开发-开发指导 文章目录 鸿蒙HarmonyOS兼容JS的类Web开发-开发指导常用组件开发指导list开发指导创建list组件添加滚动条添加侧边索引栏实现列表折叠和展开场景示例 dialog开发指导创建dialog组件设置弹窗响应场景示例 form开发指导创建form组件实现…

Python笔记08-面向对象

文章目录 类和对象构造方法内置方法封装继承类型注解多态 类只是一种程序内的“设计图纸”&#xff0c;需要基于图纸生产实体&#xff08;对象&#xff09;&#xff0c;才能正常工作 这种套路&#xff0c;称之为&#xff1a;面向对象编程 类和对象 定义类的语法如下&#xff…

Java快捷键

以下是一些常见的 Java 快捷键&#xff1a; Ctrl Shift F&#xff1a;格式化代码。Ctrl Shift O&#xff1a;导入缺失的类。Ctrl Alt L&#xff1a;格式化代码布局。**Ctrl /**&#xff1a;添加或取消单行注释。**Ctrl Shift /**&#xff1a;添加或取消多行注释。Alt …

QT上位机开发(利用tcp/ip访问plc)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 plc是工控领域很重要的一个器件。简单的plc一般就是对io进行控制&#xff0c;但是复杂的plc&#xff0c;还可以控制电机、变频器&#xff0c;在工业…

Java 数组

Array 1、java 语言中的数组是一种引用数据类型&#xff0c;不属于基本数据类型。数组的父类是 Object。 2、数组实际上是一个容器&#xff0c;可以同时容纳多个元素。&#xff08;数组是一个数据的集合。&#xff09; 数组&#xff1a;字面意思是“一组数据” 3、数组当中可以…

Python3.5如何打包编译

python3.5怎么打包编译 问题&#xff1a;用Python开发的小工具有时需要编译打包为Windows(*.exe)、Mac等操作系统下的可执行性文件以供非程序员使用。 解决方案&#xff1a; 一、py2exe 目前只支持到Python3.4&#xff0c;暂不支持Python3.5 二、PyInstaller 安装&#x…

从vue小白到高手,从一个内容管理网站开始实战开发第七天,登录功能后台功能设计--通用分页、枚举以及相关工具类

上一篇实现了数据库访问层的相关功能,还没有了解的小伙伴可以去看前面文章实现的内容,因为每一篇内容都是连贯的,不学习的话可能下面的内容学习起来会有点摸不着头脑 从vue小白到高手,从一个内容管理网站开始实战开发第六天,登录功能后台功能设计--API项目中的登录实现(二…

MongoDB聚合:$documents

$documents阶段可以根据输入值返回字面意义的文档。 语法 { $documents: <表达式> }$documents接受可解析为对象数组的任何有效表达式&#xff0c;包括&#xff1a; 系统变量&#xff0c;如 $$NOW 或 $$SEARCH_META $let 表达式 $lookup 表达式作用域中的变量 没有…

chat-plus部署指南

目录 1.下载代码 2.启动 3.测试 1.下载代码 cd /optwget https://github.com/yangjian102621/chatgpt-plus/archive/refs/tags/v3.2.4.1.tar.gz 2.启动 cd /opt/chatgpt-plus-3.2.4.1/deploydocker-compose up -d 3.测试 管理员地址xxx:8080/admin 账号密码admin/admin1…

真实可用,Xshell7 期待您的安装使用

xshell https://pan.baidu.com/s/1OKC1sQ1eYq6ZSC8Ez5s0Fg?pwd0531 1.鼠标右击【Xshell7.zip】压缩包&#xff08;win11及以上系统需先点击“显示更多选项”&#xff09; 2.双击Xshell-7.0.0065.exe 执行安装操作 3.选择【是】 4.点击【下一步】 5.选择【我接受...】 6.点击…

Open CASCADE学习|创建旋转体

旋转体是一个几何概念&#xff0c;指的是通过旋转一个平面图形得到的立体图形。具体来说&#xff0c;一个平面图形绕着它所在的平面内的一条定直线旋转一周所形成的曲面&#xff0c;这个曲面会围成一个几何体&#xff0c;这个几何体就叫做旋转体。这条定直线被称为旋转体的轴。…

DEJA_VU3D - Cesium功能集 之 113-获取圆节点(2)

前言 编写这个专栏主要目的是对工作之中基于Cesium实现过的功能进行整合,有自己琢磨实现的,也有参考其他大神后整理实现的,初步算了算现在有差不多实现小140个左右的功能,后续也会不断的追加,所以暂时打算一周2-3更的样子来更新本专栏(每篇博文都会奉上完整demo的源代码…

MySQL入门:DCL数据控制语言(管理用户,权限控制),MySQL函数(字符串,数值,日期,流程)

目录 1.DCL&#xff08;数据控制语言&#xff09;1.管理用户2.权限控制 2.函数1.字符串函数2.数值函数3.日期函数4.流程函数 1.DCL&#xff08;数据控制语言&#xff09; DCL英文全称是Data ControlLanguage(数据控制语言)&#xff0c;用来管理数据库用户、控制数据库的访问权限…