python:类 super() 的代理对象,在指定的类处启动 MRO 搜索

2023-11-29

根据文档,super(cls, obj) returns

将方法调用委托给父级或同级的代理对象 cls 类型的类

我明白为什么super()提供此功能,但我需要稍微不同的东西:我需要创建一个代理对象,将方法调用(和属性查找)委托给类cls本身;和如super, if cls没有实现方法/属性,我的代理应该继续在 MRO 订单中查找(new不是original班级)。我可以编写任何函数来实现这一目标吗?

Example:

class X:
  def act():
    #...

class Y:
  def act():
    #...

class A(X, Y):
  def act():
    #...

class B(X, Y):
  def act():
    #...

class C(A, B):
  def act():
    #...

c = C()
b = some_magic_function(B, c)
# `b` needs to delegate calls to `act` to B, and look up attribute `s` in B
# I will pass `b` somewhere else, and have no control over it

当然,我可以做b = super(A, c),但这依赖于了解确切的类层次结构and事实是B如下A在 MRO 中。如果这两个假设中的任何一个在未来发生变化,它都会悄然破裂。 (注意super不做任何这样的假设!)

如果我只需要打电话b.act(),我可以用B.act(c)。但我正在路过b给别人,并且不知道他们会用它做什么。我需要确保它不会背叛我并开始表现得像一个实例class C在某一点。

一个单独的问题,文档super()(在 Python 3.2 中)仅讨论其方法委托,并没有阐明代理的属性查找也以相同的方式执行。难道是偶然的遗漏?

EDIT

更新后的委托方法也适用于以下示例:

class A:
    def f(self):
        print('A.f')
    def h(self):
        print('A.h')
        self.f()

class B(A):
    def g(self):
        self.f()
        print('B.g')
    def f(self):
        print('B.f')
    def t(self):
        super().h()


a_true = A()
# instance of A ends up executing A.f
a_true.h()

b = B()
a_proxy = Delegate(A, b)
# *unlike* super(), the updated `Delegate` implementation would call A.f, not B.f
a_proxy.h()

请注意,更新后的class Delegate比我想要的更接近super()有两个原因:

  1. super()它只代理第一次调用;后续调用将照常发生,因为那时将使用该对象,而不是其代理。

  2. super()不允许属性访问。

因此,我所提出的问题在 Python 中有一个(几乎)完美的答案。

事实证明,在更高的层面上,我试图做一些我不应该做的事情(在这里查看我的评论).


此类应涵盖最常见的情况:

class Delegate:
    def __init__(self, cls, obj):
        self._delegate_cls = cls
        self._delegate_obj = obj
    def __getattr__(self, name):
        x = getattr(self._delegate_cls, name)
        if hasattr(x, "__get__"):
            return x.__get__(self._delegate_obj)
        return x

像这样使用它:

b = Delegate(B, c)

(使用示例代码中的名称。)

限制:

  1. 您无法检索一些特殊属性,例如__class__等等,从您通过此代理传入构造函数的类中。 (此限制也适用于super.)

  2. 如果您要检索的属性是某种奇怪的描述符,这可能会表现得很奇怪。

Edit:如果您希望问题更新中的代码按预期工作,可以使用以下代码:

class Delegate:
    def __init__(self, cls):
        self._delegate_cls = cls
    def __getattr__(self, name):
        x = getattr(self._delegate_cls, name)
        if hasattr(x, "__get__"):
            return x.__get__(self)
        return x

这将代理对象传递为self任何被调用方法的参数,并且它根本不需要原始对象,因此我从构造函数中将其删除。

如果您还希望可以访问实例属性,可以使用此版本:

class Delegate:
    def __init__(self, cls, obj):
        self._delegate_cls = cls
        self._delegate_obj = obj
    def __getattr__(self, name):
        if name in vars(self._delegate_obj):
            return getattr(self._delegate_obj, name)
        x = getattr(self._delegate_cls, name)
        if hasattr(x, "__get__"):
            return x.__get__(self)
        return x
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

python:类 super() 的代理对象,在指定的类处启动 MRO 搜索 的相关文章

  • 如何在 Python 2.4 CSV 阅读器中禁用引用?

    我正在编写一个 Python 实用程序 需要解析一个我无法控制的大型且定期更新的 CSV 文件 该实用程序必须在仅提供 Python 2 4 的服务器上运行 CSV 文件根本不引用字段值 但Python 2 4版本的csv库 http ww
  • Json.dump 失败并显示“必须是 unicode,而不是 str”TypeError

    我有一个 json 文件 其中恰好有大量中文和日文 以及其他语言 字符 我将其加载到我的 python 2 7 脚本中使用io open如下 with io open multiIdName json encoding utf 8 as j
  • 如何将 UPX 与 pyinstaller 一起使用?

    如何将 UPX 与 pyinstaller 一起使用 我正在关注文档 我已经下载了UPX 我的文件如下所示 import csv import selenium import pandas print Hello 然后我运行 pyinsta
  • python中嵌套字典值的总和

    我有一本这样的字典 data 11L a 2 b 1 a 2 b 3 22L a 3 b 2 a 2 b 5 a 4 b 2 a 1 b 5 a 1 b 0 33L a 1 b 2 a 3 b 5 a 5 b 2 a 1 b 3 a 1 b
  • 手动安装开放多语言世界网 (NLTK)

    我正在使用一台只能访问专用网络并且无法从命令行发送指令的计算机 因此 每当我必须安装 Python 包时 我都必须手动安装 我什至不能使用 Pypi 幸运的是 NLTK 允许我手动下载语料库 从here https www nltk org
  • 完全定制的Python帮助用法

    我正在尝试使用 Python 创建完全自定义的 帮助 用法 我计划将其导入到许多我想要具有风格一致性的程序中 但遇到了一些麻烦 我不知道为什么我的描述忽略换行符 尝试过 和 我无法让 出现在 ARGS 行的 换行符之后 显然它们坐在自己的行
  • 如何使直方图列的宽度都相同

    我在操作直方图时遇到了一些麻烦 我有一个包含两列的 df 我将它们绘制为堆叠直方图 我将它们放入特定的垃圾箱中 请参阅下面的代码 但我想在最后制作一个大垃圾箱 4000 10000 但是 默认情况下 大垃圾箱的列宽很大 有没有办法让这个大垃
  • 将项目导入 Eclipse 后出现“必须重写超类方法”错误

    任何时候我必须将我的项目重新导入到 Eclipse 中 如果我重新安装了 Eclipse 或者更改了项目的位置 几乎全部我的重写方法的格式不正确 导致错误 该方法必须重写超类方法 值得注意的是 无论出于何种原因 Android 项目中方法参
  • 模拟类:Mock() 还是 patch()?

    我在用mock http www voidspace org uk python mock index html使用Python 想知道这两种方法中哪一种更好 阅读 更Pythonic 方法一 只需创建一个模拟对象并使用它 代码如下 def
  • 如何向 RetrievalQA.from_chain_type 添加内存?或者,如何向 ConversationalRetrievalChain 添加自定义提示?

    如何向 RetrievalQA from chain type 添加内存 或者 如何向 ConversationalRetrievalChain 添加自定义提示 在过去的两周里 我一直在尝试制作一个可以通过文档聊天的聊天机器人 因此不仅仅是
  • Tkinter 如何根据此组合框自动更新第二个组合框

    我在 Tkinter Python 中遇到了组合框更新的问题 我有两个组合框 组合框A with values A B C and 组合框B 我想要的是 当值A在组合框中选择A然后在组合框中B显示值 1 2 3 当值B在组合框中选择A然后在
  • 如何使用 selenium 获取 javascript 结果?

    我有以下代码 from selenium import selenium selenium selenium localhost 4444 chrome http some site com selenium start sel selen
  • Python itertools groupby 中令人不安的奇怪行为/错误?

    我在用itertools groupby解析一个短的制表符分隔的文本文件 文本文件有几列 我想做的就是对具有特定值的所有条目进行分组x在特定的列中 下面的代码对名为的列执行此操作name2 寻找变量中的值x 我尝试使用以下方法来做到这一点c
  • 如何连接多个字符串? [复制]

    这个问题在这里已经有答案了 如何将 stringList 中的所有字符串合并为一个而不打印它 例如 s joinStrings very hot day returns string print s Veryhotday 感觉有点倒退 但是
  • Python 包安装:pip 与 yum,还是两者一起安装?

    我刚刚开始管理 Hadoop 集群 我们使用 Bright Cluster Manager 直至操作系统级别 CentOS 7 1 然后使用 Ambari 以及适用于 Hadoop 的 Hortonworks HDP 2 3 我不断收到安装
  • 对 Python 列表元素进行分组

    我有一个 python 列表 如下所示 my list 25 1 0 65 25 3 0 63 25 2 0 62 50 3 0 65 50 2 0 63 50 1 0 62 我想根据以下规则对它们进行排序 1 gt 0 65 0 62 l
  • 在 Python 中伪造一个对象是否是类的实例

    假设我有一堂课FakePerson它模仿基类的所有属性和功能RealPerson 不扩展它 在Python 3中 是否可以伪造isinstance 为了认识到FakePerson as a RealPerson只通过修改对象FakePers
  • python 相当于 sed

    有没有一种方法 无需双循环即可完成以下 sed 命令的操作 Input Time Banana spinach turkey sed i Banana s Toothpaste file Output Time BananaToothpas
  • 混合两个列表的Pythonic方法[重复]

    这个问题在这里已经有答案了 我有两个长度为 n 和 n 1 的列表 a 1 a 2 a n b 1 b 2 b n 1 我想要一个函数作为结果给出一个列表 其中包含两个中的替代元素 即 b 1 a 1 b n a n b n 1 以下方法有
  • Python Web 编程的不同方法的优缺点

    我想使用 Python 编写一些服务器端脚本 但我对这样做的方法有点迷失了 它从 DIY CGI 方法开始 似乎以一些相当强大的框架结束 这些框架基本上可以自己完成所有工作 中间有很多东西 比如web py http webpy org P

随机推荐

  • Python 3.4 在生成一些(但不是全部)带有分段错误 11 的 Cartopy 地图时崩溃

    我已经在运行 El Capitan 10 11 6 和 Python 3 4 的 Mac 上安装了 Python 映射工具 Cartopy 我可以使用 Cartopy 成功绘制一些地图 但在某些情况下 Python 内核会因分段错误 11
  • 无法实例化类型 [简单类型,类 java.time.LocalDate] 的值

    这里我使用 java 1 8 的 LocalDate 类 在我的 bean 中 我将返回类型设置为 LocalDate 我发送日期格式为07 01 2017 当我试图保存时 我遇到了以下异常 org springframework web
  • 虚函数和纯虚函数之间的区别[重复]

    这个问题在这里已经有答案了 纯虚函数和虚函数有什么区别 我知道 纯虚函数是一个没有函数体的虚函数 但这是什么意思以及下面的行实际上做了什么 virtual void virtualfunctioname 0 虚函数使其类成为多态基类 派生类
  • 为什么这个分支会破坏类型推断?

    我正在使用一个自行开发的实现Either在Java中 有这样的方法 public static
  • 嵌套的弹性元素不会使父元素增长

    我正在尝试编写一个标题 其中包含月份名称以及该月名称正下方的所有日期 天项目具有固定宽度 所以我想 为什么不使用 Flexbox 由于某些原因 day其父级元素正在下降 并且父级元素不会增长 你能告诉我为什么吗flex basis auto
  • 如何使用Python和Cypher在Apache Age中实现最短路径算法?

    我正在开发一个使用 Apache Age 作为图形数据库的 Python 项目 我需要找到两个可能节点之间的最短路径 我如何使用 Python 来实现这个 创建节点和图结构的代码如下 Python code to create nodes
  • 如何创建在设备和佩戴上不同的通知?

    基本上 我想知道是否有可能创建两个不同的通知以及如何创建 一个用于 Android 设备 另一个用于 Android Wear 例如 我想要setContentText 但在 Android 设备上我想要setContentTitle an
  • 在 ObservableList 上调用clear()会导致IndexOutOfBoundsException

    我有一个带有可观察列表的组合框 该列表会随着用户输入字符或进行选择而更新 我遇到的问题是当我从 ComboBox 中选择一个项目并调用侦听器事件 然后调用 ComboBox 的 ObservableList 中的clear 方法时引起的 完
  • Android TCP 应用程序挂在 inStream.readline() 上

    这是一个延续这个问题因为它回答了我原来的问题 但它没有解决错误 问题 如何修复挂在这一行的代码inStream readline 我的意图 这是在一个线程中 该线程将循环检查是否有 outMessage 如果有 它将发送消息 接下来 它将检
  • 如何在 Vim 中实现类似 MATLAB 的单元格模式

    在 MATLAB 中 我们可以向编辑器写入以下内容 a example cell plot rand 3 another cell A rand 2 10 t linspace pi pi 2 10 compass fft A sin t
  • 将 Parceler (@Parcel) 与 Realm.io (Android) 一起使用

    我有以下代码会产生错误 Error Parceler Unable to find read write generator for type io realm Realm for io realm RealmObject realm 一切
  • MATLAB 中的稳定 AccumArray

    MATLAB的内置函数accumarray接受一个函数fun作为第四个参数 A accumarray subs val sz fun 这适用fun到每个元素子集val具有相同的下标subs 然而 该文档指出 如果下标在subs未根据其线性索
  • 在下拉列表内的树层次结构中显示类别/子类别

    我有一个类别表 其中包含字段 id name 和parent id 根类别的parent id 0 现在我想在下拉列表中显示类别列表 结构如下 root category first sub category sub sub categor
  • .NET 4 GUID 是如何生成的?

    我知道有很多问题here以及雷蒙德的出色 像往常一样 post 然而 由于创建 GUID 的算法明显发生了变化 我发现很难获得任何最新信息 这MSDN似乎试图提供尽可能少的信息 关于 NET 4 中如何生成 GUID 我们了解多少 更改了什
  • ID 保存并显示名称 - 带芯片的材质 ui 多选

    我正在使用基于文档示例的 Material UI Multiple Select 我需要保存所选选项的 ID 和显示名称 我尝试了很多方法但没有任何帮助 我需要存储 id 并显示值 请帮助我指导 这是我的代码和盒子link 我也想知道同样的
  • 奇怪的谷歌地图行为 - 瓷砖仅出现在左上象限

    使用谷歌地图数十次 但无法绕过这一点 为地图版本 3 执行 google 的简单 hello world http code google com apis maps documentation javascript tutorial ht
  • 赋值关联性[重复]

    这个问题在这里已经有答案了 赋值运算符有右到左关联性 所以 var x y x y 1 按预期工作 x 等于 1 现在考虑代码 var foo foo x foo n 2 我希望上面的工作像下面这样 var foo n 2 foo x fo
  • 管理两个 NSURLConnection

    我想从两个不同的 kml 文件执行两个异步请求 因此我首先设置两个请求 NSString server1URL NSString stringWithFormat NSMutableURLRequest firstRequest NSMut
  • Github 操作在作业之间共享工作空间/工件?

    尝试使用 Github 的 beta 操作时 我有两项工作 一项是构建代码 另一项是部署代码 但是 我似乎无法在部署作业中获取构建工件 我最新的尝试是为每个作业手动设置具有相同卷的容器映像 根据文档 这应该是解决方案 https help
  • python:类 super() 的代理对象,在指定的类处启动 MRO 搜索

    根据文档 super cls obj returns 将方法调用委托给父级或同级的代理对象 cls 类型的类 我明白为什么super 提供此功能 但我需要稍微不同的东西 我需要创建一个代理对象 将方法调用 和属性查找 委托给类cls本身 和