3

我有一些我在 python 2.4++ 中开发的代码,你敢打赌,我必须将它反向移植到 python 2.1!

函数装饰器非常吸引人,以至于我在几个地方使用了@classmethod,但没有注意到它仅从 2.4 版开始可用。函数修饰符提供了相同的功能,但这些出现在 python 2.2 中。cfr: http: //www.python.org/dev/peps/pep-0318/

现在我们的一些客户似乎仍然依赖于 python 2.1(ArcGIS 9.1 附带它并使其不可升级),甚至函数修饰符都不可用......

我一直在寻找 python 2.1 中的一些函数修饰符定义,但我没有找到任何(我的意思是:工作)。有人成功解决了这个问题吗?

更具体地说,我需要一种在 python 2.1 中运行此 2.4 代码的方法:

Python 2.4.6 (#2, Mar 19 2009, 10:00:53) 
[GCC 4.3.3] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> class sic:
...   def f(cls):
...     print cls.__name__
...   f = classmethod(f)
... 
>>> sic().f()
sic
>>> sic.f()
sic
>>> 
4

2 回答 2

2

就像我在 2.1 静态方法的旧配方中一样:

class staticmethod:
    def __init__(self, thefunc): self.f = thefunc
    def __call__(self, *a, **k): return self.f(*a, **k)

您应该能够执行 2.1 类方法:

class classmethod:
    def __init__(self, thefunc): self.f = thefunc
    def __call__(self, obj, *a, **k): return self.f(obj.__class__, *a, **k)

没有@-syntax 当然,而是旧的方式(如在 2.2 中):

class sic:
  def f(cls): ...
  f = classmethod(f)

如果这不起作用(对不起,自从我有一个 Python 2.1 测试以来已经很多年了),需要更明确地提供该类 - 并且由于您在类对象存在之前调用 classmethod,它需要按名称命名——假设是一个全局类,

class classmethod2:
    def __init__(self, thefunc, clsnam):
        self.f = thefunc
        self.clsnam = clsnam
    def __call__(self, *a, **k):
        klass = globals()[self.clsnam]
        return self.f(klass, *a, **k)

class sic2:
  def f(cls): ...
  f = classmethod2(f, 'sic2')

很难找到优雅的方法来获取类对象(抑制对 self 的需要是容易的部分,而静态方法就足够了:你只需要将函数包装成非函数可调用),因为 2.1 只有遗留(旧-style 类),没有可用的元类,因此没有真正好的方法让 sic2.f() 神奇地获得那个 cls。

由于 2.1 中缺少 @ 语法不可避免地需要编辑使用 的代码,因此@classmethod另一种方法是将功能(将一些方法装饰为“类”方法)在class语句结束后向右移动(优点是类对象当时确实存在)。

class classmethod3:
    def __init__(self, thefunc, klass):
        self.f = thefunc
        self.klass = klass
    def __call__(self, *a, **k):
        return self.f(self.klass, *a, **k)

def decorate(klass, klassmethodnames):
  for n in klassmethodnames:
    thefunc = klass.__dict__[n]
    setattr(klass, n, classmethod3(thefunc, klass))

class sic2:
  def f(cls): ...
  def g(self): ...
  def h(cls): ...
decorate(sic2, ['f', 'h'])
于 2009-07-21T14:12:57.357 回答
1

如果只需要将 @classmethod 向后移植到 Python 2.1。
python2.1中有一个Classmethod emulation的配方
希望对您有所帮助。

于 2009-07-21T13:22:43.083 回答