我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Python的代理类怎么实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Python的代理类怎么实现

这篇“Python的代理类怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Python的代理类怎么实现”文章吧。

代理类的一个简单的实现方式示例

目标:实现类Product的实例属性让另一个类Proxy来代理访问和控制,想将对外公布的属性交给代理类让外部访问和控制,不想对外公布的属性无法通过代理来访问和控制,这里不想对外公布的属性约定用下划线命名开头

# proxy_example1.py# 以下是一个代理类实现只读访问的示例# 目标:代理后只能访问和修改Product的公开属性,私有属性_current只能查看不能修改class Product:    def __init__(self, price, quantity):        self.price = price        self.quantity = quantity        self._current = 123# 只暴露代理类Proxy给外部使用class Proxy:    def __init__(self, obj):        self._obj = obj    def __getattr__(self, item):    # 本实例没有找到的属性会执行__getattr__方法        if item.startswith("_"):    # 约定下划线开头的方法不能访问到被代理的类,只会访问到代理类            raise Exception(f"{item} not found")    # Product存在的私有属性也不希望被外部知道        return getattr(self._obj, item)    def __setattr__(self, key, value):        if key.startswith("_"):     # 约定下划线开头的方法不能访问到被代理的类,只会访问到代理类            # 注:这里不能raise,这会导致Proxy的实例都无法创建(__dict__等属性无法创建)            super(Proxy, self).__setattr__(key, value)   # 避免无限循环        else:            setattr(self._obj, key, value)    # 要求只能删除非下划线开头的属性    def __delattr__(self, item):        if item.startswith("_"):            super(Proxy, self).__delattr__(item)    # 避免无限循环        else:            delattr(self._obj, item)def test_getattr():    p = Product(10, 1)    pp = Proxy(p)    print(pp.price)    print(pp._curr)def test_setattr():    p = Product(10, 2)    pp = Proxy(p)    pp.abc = 1    print(pp.abc, p.abc)    pp._curr = 10000    print(pp._curr)  # 私有属性,设置给了代理类    print(p._curr)  # raise an error, 被代理的类Product的属性没有设置成功也无法访问def test_delattr():    p = Product(10, 2)    pp = Proxy(p)    pp.abc = 123    print(pp.abc, p.abc)    # 删除公开属性    del pp.abc  # 成功    # print(pp.abc, p.abc)  # 已被删除    # # 删除私有属性    # del pp._curr    # 会尝试删除Proxy的私有属性,raise AttributeError: _curr    # 先创建在删除    pp._def = 123   # 这个操作只会设置Proxy的实例属性    print(pp._def)      # 访问的是Proxy实例属性,被代理的Product实例没有创建_def属性    # del pp._def     # 删除的是Proxy的实例属性    # print(pp._def)

测试获取属性

if __name__ == '__main__':    test_getattr()

输出:

10
...
Exception: _curr not found
...

测试设置属性

if __name__ == '__main__':    test_delattr()

输出

1 1
10000
...
AttributeError: 'Product' object has no attribute '_curr'
...

测试删除属性

if __name__ == '__main__':    test_delattr()

输出

123 123
123

注:以双下划线开头和结尾的方法无法被代理,想要使用,必须在代理类中定义出这个方法,然后重定向到被代理的类的方法,比如你想使用isinstance()方法就要在Proxy伪造定义__class__属性,想要使用len()方法就要在Proxy重定向返回到被代理的类的len方法

# proxy_example2.pyclass Product:    def __init__(self, price, quantity):        self.price = price        self.quantity = quantity        self._current = 123    def __len__(self):        return 111# 只暴露代理类Proxy给外部使用class Proxy:    def __init__(self, obj):        self._obj = obj    def __getattr__(self, item):    # 本实例没有找到的属性会执行__getattr__方法        if item.startswith("_"):    # 约定下划线开头的方法不能访问到被代理的类,只会访问到代理类            raise Exception(f"{item} not found")    # Product存在的私有属性也不希望被外部知道        return getattr(self._obj, item)    def __setattr__(self, key, value):        if key.startswith("_"):     # 约定下划线开头的方法不能访问到被代理的类,只会访问到代理类            # 注:这里不能raise,这会导致Proxy的实例都无法创建(__dict__等属性无法创建)            super(Proxy, self).__setattr__(key, value)   # 避免无限循环        else:            setattr(self._obj, key, value)    # 要求只能删除非下划线开头的属性    def __delattr__(self, item):        if item.startswith("_"):            super(Proxy, self).__delattr__(item)    # 避免无限循环        else:            delattr(self._obj, item)    @property    def __class__(self):    # 伪造类        return self._obj.__class__    def __len__(self):        return len(self._obj)def test_instance():    p = Product(10, 2)    pp = Proxy(p)    print(pp.__class__)    print(isinstance(pp, Product))      # 如果不伪造__class__,会返回Falsedef test_len():    p = Product(10, 2)    pp = Proxy(p)    print(len(pp))  # 如果Proxy实例不定义__len__方法,会报错TypeError: object of type 'Proxy' has no len()

测试伪造的实例class类型

if __name__ == '__main__':    test_instance()

输出

<class '__main__.Product'>
True

测试获取长度

if __name__ == '__main__':    test_len()

输出

111

一个实现日志输出的代理类的简化示例

捕获web server报错日志并执行异常处理的示例

# logger_proxy.py# -*- coding:utf-8 -*-from functools import wrapsclass DAL:    @classmethod    def dm1(cls, req, *args):        print("dm1...", f"{req=}")        print(1/0)      # 故意抛出异常        return "dm1"class BLL:    @classmethod    def bm1(cls, req):        print("bm1...", f"{req=}")        return DAL.dm1(req)class Application:    def __init__(self, req):        self.req = req        self._p = "private attr"    def hd1(self):        return BLL.bm1(self.req)class LoggerProxy:    def __init__(self, obj):        self._obj = obj    def __getattr__(self, item):    # LoggerProxy类实例没获取到的属性会执行这个方法        attr = getattr(self._obj, item)        if callable(attr):  # 获取到了方法,则处理异常捕获            @wraps(attr)            def wrapped_method(*args, **kwargs):                # print(f"Before access to attribute/method: {item}")                try:                    method = attr(*args, **kwargs)                except ZeroDivisionError:                    # 捕获异常然后处理...                    raise Exception(f"{attr.__name__} received a zero division error.")                # print(f"After attribute/method {item} returned")                return method            return wrapped_method        else:   # 获取到了属性,直接返回            return attrif __name__ == '__main__':    lp = LoggerProxy(Application("abc"))    print(lp.req)    print(lp._p)    print(lp.hd1())

运行输出

abc
private attr
bm1... req='abc'
dm1... req='abc'
Traceback...
ZeroDivisionError: division by zero
During handling of the above exception, another exception occurred:
Traceback...
Exception: hd1 received a zero division error.

以上就是关于“Python的代理类怎么实现”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Python的代理类怎么实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Python的代理类怎么实现

这篇“Python的代理类怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Python的代理类怎么实现”文章吧。代理类
2023-06-29

python中怎么利用class类实现可迭代

今天就跟大家聊聊有关python中怎么利用class类实现可迭代,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。1. 当定义一个普通的类时,指向类的实例默认情况下是不可迭代的,如下In
2023-06-01

python上下文管理器怎么实现类

本篇内容介绍了“python上下文管理器怎么实现类”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!说明1、通过定义__enter__和__ex
2023-06-20

Python中类的继承怎么实现

本篇内容主要讲解“Python中类的继承怎么实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Python中类的继承怎么实现”吧!1. 单继承继承的主要作用是实现代码的重用。继承使得子类拥有父类
2023-07-02

python实现图书管理系统的代码怎么写

这篇文章主要介绍“python实现图书管理系统的代码怎么写”,在日常操作中,相信很多人在python实现图书管理系统的代码怎么写问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”python实现图书管理系统的代码
2023-06-29

Python爬虫怎么实现搭建代理ip池

这篇“Python爬虫怎么实现搭建代理ip池”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Python爬虫怎么实现搭建代理i
2023-07-02

python 实现 代理模式

本文的目录地址本文的代码地址在某些应用中,我们需要在访问某个对象之前执行一个或多个重要的操作,例如,访问敏感信息——在允许用户访问敏感信息之前,我们希望确保用户具备足够的权限。操作系统也存在类似的情况,用户必须具有管理员权限才能在系统中安装
2023-01-31

Python实现的异步代理爬虫及代理池

使用python asyncio实现了一个异步代理池,根据规则爬取代理网站上的免费代理,在验证其有效后存入redis中,定期扩展代理的数量并检验池中代理的有效性,移除失效的代理。同时用aiohttp实现了一个server,其他的程序可以通过
2022-06-04

怎么用一行代码实现Python并行处理

今天小编给大家分享一下怎么用一行代码实现Python并行处理的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。传统的例子简单搜索
2023-06-27

简单实现Socks5代理(Python&

只实现了CONNECT功能参考:点击打开链接Python版:import socketimport threadingimport selectimport timeIsNeedAuth=FalseUsername='admin'Passw
2023-01-31

怎么在Python中实现类的单继承

怎么在Python中实现类的单继承?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。一、类的继承面向对象三要素之一,继承Inheritance人类和猫类都继承自动物类。个体继承自父
2023-06-15

python数据处理之Pandas类型转换怎么实现

这篇文章主要介绍“python数据处理之Pandas类型转换怎么实现”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“python数据处理之Pandas类型转换怎么实现”文章能帮助大家解决问题。转换为字
2023-06-30

怎么实现socks5全局代理

要实现Socks5全局代理,你可以按照以下步骤进行操作:1. 首先,你需要一个Socks5代理服务器。你可以使用开源的Socks5服务器软件,如Shadowsocks、Squid等,或者购买一个Socks5代理服务。2. 配置代理服务器。根
2023-08-19

Java代理模式怎么实现

这篇文章主要讲解了“Java代理模式怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java代理模式怎么实现”吧!结构型模式前面创建型模式介绍了创建对象的一些设计模式,这节介绍的结构型
2023-06-02

Go代理模式怎么实现

Go语言中实现代理模式可以使用结构体和接口的方式。首先,定义一个接口,代理和被代理对象都必须实现这个接口。这里以一个简单的计算器为例:```gotype Calculator interface {Add(a, b int) intSubt
2023-08-08

Python实现图书管理系统设计的代码怎么写

本篇内容介绍了“Python实现图书管理系统设计的代码怎么写”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!基于python的tkinter,
2023-06-29

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录