6

次のようなことができます。

class master:
    @combomethod 
    def foo(param):
        param.bar() # Param could be type as well as object

class slaveClass( master ):
    @classmethod 
    def bar(cls):
        print("This is class method")

slaveType = slaveClass
slaveType.foo()

class slaveInstance( master ):
    def __init__(self, data):
        self.data = data
    def bar(self):
        print("This is "+self.data+" method")


slaveType = slaveInstance("instance")
slaveType.foo()

combomethodは、「インスタンスとクラス メソッドを兼ねるメソッドの作成」で定義されています。

私の質問は、デフォルトの最初のパラメーターをコンボクラスのパラメーターとして使用できないのはなぜですか? または、少なくとも、最初のパラメーターとしてオブジェクトを classmethod に渡すことができないのはなぜですか? クラスメソッドとインスタンスメソッドの違いは知っていますし、デコレーターも知っていますが、組み込み@classmethodselfパラメーターの受け渡しがどのように行われるかは理解していないかもしれません。技術的な制限はありますか? または、なぜまだcombomethod組み込まれていないのですか?

4

2 に答える 2

10

これを使って:

class A(object):

    @classmethod
    def print(cls):
        print 'A'

    def __print(self):
        print 'B'

    def __init__(self):
        self.print = self.__print


a = A()
a.print()
A.print()
于 2013-02-01T09:04:25.050 に答える
4

combomethodアクセス時にメソッドオブジェクトを作成しませんが、特別にラップされた関数を作成します。メソッドと同様に、アクセスごとに新しいオブジェクト (この場合は新しい関数オブジェクト) が作成されます。

class A:
    def __init__(self):
        self.data = 'instance'

    @combomethod 
    def foo(param):
        if isinstance(param, A):
            print("This is an " + param.data + " method.")
        elif param is A:
            print("This is a class method.")

>>> a = A()
>>> A.foo
<function foo at 0x00CFE810>
>>> a.foo
<function foo at 0x00CFE858>

>>> A.foo()
This is a class method.
>>> a.foo()
This is an instance method.

アクセスごとに新しい:

>>> A.foo is A.foo
False
>>> a.foo is a.foo
False

foo本当に_wrapper変装しています:

>>> A.foo.__code__.co_name
'_wrapper'

クラスから呼び出された場合、クロージャーは obj == None を持ちます (ここでの「自己」は、元の関数オブジェクトへの参照を持つコンボメソッドを指すことに注意してくださいself.method):

>>> print(*zip(A.foo.__code__.co_freevars, A.foo.__closure__), sep='\n')
('obj', <cell at 0x011983F0: NoneType object at 0x1E1DF8F4>)
('self', <cell at 0x01198530: combomethod object at 0x00D29630>)
('objtype', <cell at 0x00D29D10: type object at 0x01196858>)

インスタンスの属性として呼び出されると、obj がインスタンスになります。

>>> print(*zip(a.foo.__code__.co_freevars, a.foo.__closure__), sep='\n')
('obj', <cell at 0x01198570: A object at 0x00D29FD0>)
('self', <cell at 0x01198530: combomethod object at 0x00D29630>)
('objtype', <cell at 0x00D29D10: type object at 0x01196858>)

以下は、combomethod に格納されている元の関数です。

>>> A.foo.__closure__[1].cell_contents.method
<function foo at 0x00D1CB70>
>>> A.foo.__closure__[1].cell_contents.method.__code__.co_name
'foo'

_wrapperself.methodobj の値を指定して、クラスまたはインスタンスのいずれかを最初の引数として実行します。

if obj is not None:
    return self.method(obj, *args, **kwargs)
else:
    return self.method(objtype, *args, **kwargs)
于 2011-09-10T20:15:39.253 に答える