9

クラスのメソッドからモジュールレベルの関数を動的に作成しようとしています。したがって、クラス内のすべてのメソッドについて、クラスのインスタンスを作成してからメソッドを呼び出す同じ名前の関数を作成したいと思います。

これを実行したい理由は、Fabricファイルを作成するためにオブジェクト指向のアプローチを取ることができるようにするためです。Fabricはモジュールレベルの関数を呼び出しますが、クラスのメソッドは呼び出さないため、これが私の回避策です。

私は私を始めるために以下のリンクを使用しました

そして、私は次のコードを思いついた

import inspect
import sys
import types

class TestClass(object):
    def __init__(self):
        pass

    def method1(self, arg1):
        print 'method 1 %s' % arg1

    def method2(self):
        print 'method 2'

def fabric_class_to_function_magic(module_name):
    # get the module as an object
    print module_name
    module_obj = sys.modules[module_name]
    print dir(module_obj)

    # Iterate over the methods of the class and dynamically create a function
    # for each method that calls the method and add it to the current module
    for method in inspect.getmembers(TestClass, predicate=inspect.ismethod):
        print
        print method
        method_name, method_obj = method

        # create a new template function which calls the method
        def newfunc_template(*args, **kwargs):
            tc = TestClass()
            func = getattr(tc, method_name)
            return func(*args, **kwargs)

        # create the actual function
        print 'code: ', newfunc_template.func_code
        print 'method_name: ', method_name
        newfunc = types.FunctionType(newfunc_template.func_code,
                                     {'TestClass': TestClass,
                                      'getattr': getattr,
                                      'method_name': method_name,
                                      },
                                     name=method_name,
                                     argdefs=newfunc_template.func_defaults,
                                     closure=newfunc_template.func_closure,
                                     )

        # add the new function to the current module
        setattr(module_obj, method_name, newfunc)

# test the dynamically created module level function
thismodule = sys.modules[__name__]
print dir(thismodule)
fabric_class_to_function_magic(__name__)
print dir(thismodule)
method1('arg1')
method2()

そして、私は次のエラーを受け取ります

['TestClass', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'fabric_class_to_function_magic', 'inspect', 'sys', 'thismodule', 'types']
__main__
['TestClass', '__builtins__', '__doc__', '__file__', '__name__', '__package__', 'fabric_class_to_function_magic', 'inspect', 'sys', 'thismodule', 'types']

('__init__', <unbound method TestClass.__init__>)
code:  <code object newfunc_template at 0x7f8800a28d50, file "test.py", line 85>
method_name:  __init__

('method1', <unbound method TestClass.method1>)
code:  <code object newfunc_template at 0x7f8800a28d50, file "test.py", line 85>
method_name:  method1

('method2', <unbound method TestClass.method2>)
code:  <code object newfunc_template at 0x7f8800a28d50, file "test.py", line 85>
method_name:  method2
['TestClass', '__builtins__', '__doc__', '__file__', '__init__', '__name__', '__package__', 'fabric_class_to_function_magic', 'inspect', 'method1', 'method2', 'sys', 'thismodule', 'types']
Traceback (most recent call last):
  File "test.py", line 111, in <module>
    method1('arg1')
  File "test.py", line 88, in newfunc_template
    return func(*args, **kwargs)
TypeError: method2() takes exactly 1 argument (2 given)

関数への参照を再利用しているようですか?何か案は?

更新:NedBatchelderの修正を含む作業コードは次のとおりです

def fabric_class_to_function_magic(module_name):
    # get the module as an object
    module_obj = sys.modules[module_name]

    # Iterate over the methods of the class and dynamically create a function
    # for each method that calls the method and add it to the current module
    for method in inspect.getmembers(TestClass, predicate=inspect.ismethod):
        method_name, method_obj = method

        # get the bound method
        tc = TestClass()
        func = getattr(tc, method_name)

        # add the function to the current module
        setattr(module_obj, method_name, func)

更新2:この件に関する私のブログ投稿は次のとおりです:http://www.saltycrane.com/blog/2010/09/class-based-fabric-scripts-metaprogramming-hack/

4

2 に答える 2

9

あなたは解決策を考えすぎています。の末尾を次のように変更fabric_class_to_function_magicします。

    tc = TestClass()
    func = getattr(tc, method_name)

    # add the new function to the current module
    setattr(module_obj, method_name, func)

そしてそれはうまくいきます。新しい関数オブジェクトを作成する必要はありません。オブジェクトの getattr によって返された関数オブジェクトが既にあります。getattr によって返されるバインドされたメソッドは、呼び出し可能なものです。それをモジュール属性に割り当てるだけで、準備完了です。

于 2010-09-08T03:12:22.073 に答える
1

実際にはあなたのコードは正しいですが、 return func(*args, **kwargs) が実行されると、args は () のような空のタプルを渡し、method2 にはパラメーターがないため、そのような例外が発生します。

あなたの問題に対する迅速な解決策は、次のようになります

class TestClass(object):
    def __init__(self):
        pass

    def method1(self, arg1):
        print 'method 1 %s' % arg1

    def method2(self, *args, **kw):
        print 'method 2'
于 2010-09-08T03:04:26.090 に答える