作用域 - 我想在python用exec动态执行某些语句,动态定义一个类并随后调用这个类,但是出现了问题。
ringa_lee
ringa_lee 2017-04-18 10:05:37
0
2
627

问题的描述

现在有一个需求,就是python中使用字符串动态定义一个类,并随后使用其中的方法。

可以运行的代码

# /test.py
# coding=utf-8

content = '''
class MyClass:
    def __init__(self):
        self.name = None
        self.age = None
def do():
    return MyClass()
'''

exec content
print do()
# 或者最后一句话改成exec("print do()")

直接运行这段代码是没有问题的,得到了输出<__main__.MyClass instance at 0x000000000243EB88>

不可以运行的代码

首先定义另一个actor.py文件:

# /actor.py
# coding=utf-8

def execute(content):
    exec content
    return do()

然后定义test.py文件:

# /test.py
# coding=utf-8

import actor

content = """
class MyClass:
    def __init__(self):
        self.name = None
        self.age = None
def do():
    return MyClass()
"""
print actor.execute(content)

运行test.py文件,会出现NameError: global name 'MyClass' is not defined
我的需求就是,定义一个模块,在这个模块的函数中执行一段指定的字符串,动态定义一个类,并且需要调用这个类,现在遇到的问题如上所示,求助啊。。。

ringa_lee
ringa_lee

ringa_lee

全部回覆(2)
Peter_Zhu

首先「exec」是不被推薦的方法,因為它會帶來一些問題:

  • 一些基於__module__屬性的模組會失敗,例如pickle,inspect,pydoc等

  • 記憶體洩漏

  • namespace和module shutdown behavior issue

關於這幾點問題的詳細描述,可以參考:http://lucumr.pocoo.org/2011/...

既然你硬要這麼做的話,下面程式碼可以提供一點參考:

  • tester.py

# encoding: utf-8
# tester.py

import actor

content = """
class MyClass:
    def __init__(self):
        self.name = None
        self.age = None

def do():
    return MyClass()
"""

vars = {}
code = compile(content, '<string>', 'exec')
m_cls = actor.execute(code, vars, vars)
print m_cls.name
  • actor.py

# encoding: utf-8
# actor.py

def execute(content, m_globals, m_locals):
    exec(content, m_globals, m_locals)
    return m_globals['do']()
大家讲道理

動態建立一個類別

def init(self, name):
    self.name = name

attrs = dict(name=None, age=None, __init__=init )
# type创建一个类, base object, 设置属性和__init__等
klass = type("MyClass", (object, ), attrs)

my = klass(name="Hello")

print(my)       # <__main__.MyClass object at 0x10b882290>
print(my.name)  # Hello

你會考慮用exec, 可能是沒接觸到python的高階特性

熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板