- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
作为一个简单的例子,拿一个类 Polynomial
class Polynomial(object):
def __init__(self, coefficients):
self.coefficients = coefficients
对于形式为 p(x) = a_0 + a_1*x + a_2*x^2 + ... + a_n*x^n
的多项式,其中列表 系数 = ( a_0, a_1, ..., a_n)
存储这些系数。
然后,一个插件模块 horner
可以提供函数 horner.evaluate_polynomial(p, x)
来评估 Polynomial
实例 p
at value x
,即返回p(x)
的值。但是,与其以这种方式调用函数,不如调用 p.evaluate(x)
(或更直观地通过 __call__
调用 p(x)
)会更好。但是应该怎么做呢?
a) 猴子修补,即
Polynomial.evaluate = horner.evaluate_polynomial
# or Polynomial.__call__ = horner.evaluate_polynomial
b) 子类化和替换类,即
orgPolynomial = Polynomial
class EvaluatablePolynomial(Polynomial):
def evaluate(self, x):
return horner.evaluate_polynomial(self, x)
Polynomial = EvaluatablePolynomial
c) 混合+替换,即
orgPolynomial = Polynomial
class Evaluatable(object):
def evaluate(self, x):
return horner.evaluate_polynomial(self, x)
class EvaluatablePolynomial(Polynomial, Evaluatable):
pass
Polynomial = EvaluatablePolynomial
果然,猴子修补是最短的(特别是因为我没有包含任何检查 à la hasattr(Polynomial, 'evaluate')
,但类似地子类应该调用 super()
then...),但它是最 Pythonic 的吗?或者有其他更好的选择吗?
特别是考虑到多个插件提供相同功能的可能性,例如zeros
使用 numpy
或自制二分法,当然应该只使用一个实现插件,哪种选择可能更不容易出错?
最佳答案
将函数直接修补到原始类而不是替换它的一个可能也是最重要的属性是,在加载插件之前对原始类的引用/实例现在也将具有新属性。在给定的示例中,这很可能是所需的行为,因此应该使用。
然而,在其他情况下,猴子补丁可能会以与其原始实现不兼容的方式修改现有方法的行为,并且修改后的类的先前实例应使用原始实现。诚然,这不仅很少见,而且是糟糕的设计,但您应该牢记这种可能性。由于某些令人费解的原因,代码甚至可能依赖 不存在 猴子补丁添加方法,尽管在这里似乎很难想出一个非人工示例。
总而言之,除了少数异常(exception),将方法猴子修补到原始类中(最好在修补之前使用 hasattr(...)
检查)应该是首选方法。
编辑 我目前的做法是:创建一个子类(用于更简单的代码完成和修补),然后使用以下 patch(patching_class, unpatched_class)
方法:
import logging
from types import FunctionType, MethodType
logger = logging.getLogger(__name__)
applied_patches = []
class PatchingError(Exception):
pass
def patch(subcls, cls):
if not subcls in applied_patches:
logger.info("Monkeypatching %s methods into %s", subcls, cls)
for methodname in subcls.__dict__:
if methodname.startswith('_'):
logger.debug('Skipping methodname %s', methodname)
continue
# TODO treat modified init
elif hasattr(cls, methodname):
raise PatchingError(
"%s alrady has methodname %s, cannot overwrite!",
cls, methodname)
else:
method = getattr(subcls, methodname)
logger.debug("Adding %s %s", type(method), methodname)
method = get_raw_method(methodname, method)
setattr(cls, methodname, method)
applied_patches.append(subcls)
def get_raw_method(methodname, method):
# The following wouldn't be necessary in Python3...
# http://stackoverflow.com/q/18701102/321973
if type(method) == FunctionType:
logger.debug("Making %s static", methodname)
method = staticmethod(method)
else:
assert type(method) == MethodType
logger.debug("Un-bounding %s", methodname)
method = method.__func__
return method
悬而未决的问题是各个子类的模块是否应该在导入时直接调用 patch
还是应该手动完成。我也在考虑为这样的修补子类编写装饰器或元类......
关于python - 插件是否应该添加新的实例方法 monkey-patch 或子类/mixin 并替换父类?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18466214/
我正在尝试使用 Android SDK 中的 Draw9Patch 工具,但在使用 Draw9Patch 打开我的图像后,它立即将我的所有图像区域显示为“坏补丁”。我手动绘制了定义可拉伸(stretc
我有一个运行补丁命令的构建工具,如果补丁命令返回非零值,它将导致构建失败。我正在应用一个可能已经或可能尚未应用的补丁,所以我使用 -N选项 patch ,它应该跳过。但是,当它确实跳过时,patch返
patching file chrome/browser/gpu_process_host_ui_shim.cc Unreversed patch detected! Skipping patch.
尝试查看哪些模型最适合api(更新少,但对象结构可能经常更改,高读取应用程序) 我有这样的资源 Epic(ID、名称、描述、开始日期、结束日期、状态、故事) 故事(ID、名称、说明、开始日期、结束日期
我创建了新问题,因为我觉得上一个问题已经得到解答,这属于一个新的地方。 我跑 bitbake core-image-minimal我收到以下错误: ERROR: systemd-1_232-r0 do
我正在使用 ColdFusion 10 与 PayPal 的服务器和 for some requests 进行通信我需要执行 HTTP PATCH 请求 are not supported by CF
我试图了解这两种模拟方法之间的区别。有人可以帮助区分它们吗?对于这个例子,我使用 passlib 库。 from passlib.context import CryptContext from un
WebTarget webTarget = httpClient.target(url); Invocation.Builder invocationBuilder = webTarget.reque
当提到“提交补丁”时,补丁这个词究竟是什么意思? 我已经看到它被大量使用,尤其是在开源世界中。它是什么意思,提交补丁到底涉及什么? 最佳答案 这是一个文件,其中列出了已更改的代码文件之间的差异。它通常
对于 matplotlib.patches,patch.contains_point(xy) 方法似乎与 patch.get_path().contains_point(xy) 不同,至少在拥有之后将
这是什么RFC 5789说: The PATCH method requests that a set of changes described in the request entity be ap
在 Draw 9-patch 中,一切看起来都很好。 但是,我的 SDK 说 9-patch png 格式不正确。因为我有类似 11-patch png 的东西。因为我不希望小抓取区域被缩放。如何让它
我创建了一个使用 javax.xml.ws.Endpoint 来创建 REST 端点的类: @WebServiceProvider @ServiceMode(value = javax.xml.ws.
语境 我有一个 spring boot (version 2.2.6.RELEASE) web 项目。 从这个 Web 应用程序(我称之为“APP1”)我想使用来自另一个 Web 应用程序的 PATC
在为我的应用程序编写单元测试时,我一直使用 @mock.patch 和 @patch.object 装饰器。但是现在,当我使用装饰器进行一些单元测试时,我收到错误消息“TypeError: stati
我在使用@mock.patch.object 函数时观察到 nosetests 的一个非常奇怪的行为: 当我同时运行多个测试时,我得到的结果与单独运行它们时不同。具体来说,在某些情况下,当我一起运行多
我正在使用 RestSharp v107。 我想更新测试用例的迭代路径。我可以使用 Postman 更新它,但使用 RestSharp 我收到“BAD Request”-“您必须在请求正文中传递有效的
我已经阅读了 GNU 项目中关于开源和其他许可证的文章。某些许可证允许您将更改发布为补丁,而不是完整的源代码(例如 Q 公共(public)许可证或 gnuplot 许可证)。这是什么意思?这样的补丁
有谁知道免费的优质补丁程序?您知道,可以将其中包含旧程序的目录放入其中,然后将其与具有新版本的目录进行比较,然后吐出一个补丁,这仅仅是两者之间的区别? 另外,我正在寻找可以修补整个目录的东西,而不仅仅
由于我当时一直在使用 Subversion 和 shell 工具,git-gui这些都是不可能的。是否有任何 shell 工具可以交互式地逐行应用补丁? 最佳答案 尝试通过 --dry-run选项 p
我是一名优秀的程序员,十分优秀!