gpt4 book ai didi

python - 获取python中类的类路径或 namespace ,即使它是嵌套的

转载 作者:太空狗 更新时间:2023-10-29 21:27:30 26 4
gpt4 key购买 nike

我目前正在用 Python 编写一个序列化模块,可以序列化用户定义的类。为此,我需要获取对象的完整 namespace 并将其写入文件。然后我可以使用该字符串重新创建对象。

例如,假设我们在名为 A.py

的文件中具有以下类结构
class B:
class C:
pass

现在假设 my_klass_string 是字符串 "A::B::C"

klasses = my_klass_string.split("::")
if globals().has_key(klasses[0]):
klass = globals()[klasses[0]]
else:
raise TypeError, "No class defined: %s} " % klasses[0]
if len(klasses) > 1:
for klass_string in klasses:
if klass.__dict__.has_key(klass_string):
klass = klass.__dict__[klass_string]
else:
raise TypeError, "No class defined: %s} " % klass_string
klass_obj = klass.__new__(klass)

我可以创建类 C 的实例,即使它位于模块 A 中的类 B 下。上面的代码相当于调用 eval(klass_obj = A.B.C.__new__(A.B.C))

注意:我在这里使用 __new__() 因为我正在重构一个序列化对象并且我不想初始化对象因为我不知道类的 __init__ 的参数是什么> 方法需要。我想在不调用 init 的情况下创建对象,然后稍后为其分配属性。

我可以通过任何方式从字符串创建类 A.B.C 的对象。回合我怎么走另一条路?如何从该类的实例获取描述该类的完整路径的字符串,即使该类是嵌套的?

最佳答案

给定类的实例,您无法获得“类的完整路径”class”,因为在 Python 中没有这样的东西。对于例如,以您的示例为基础:

>>> class B(object):
... class C(object):
... pass
...
>>> D = B.C
>>> x = D()
>>> isinstance(x, B.C)
True

x 的“类路径”应该是什么? D 还是 B.C?两者都是同样有效,因此 Python 没有给你任何告诉一个人的方法从另一个。

事实上,即使是 Python 的 pickle 模块在 pickle 对象 x 时也有问题:

>>> import pickle
>>> t = open('/tmp/x.pickle', 'w+b')
>>> pickle.dump(x, t)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/usr/lib/python2.6/pickle.py", line 1362, in dump
Pickler(file, protocol).dump(obj)
...
File "/usr/lib/python2.6/pickle.py", line 748, in save_global
(obj, module, name))
pickle.PicklingError: Can't pickle <class '__main__.C'>: it's not found as __main__.C

所以,一般来说,除了添加一个属性,我看不到其他选择到你所有的类(比如,_class_path),你的序列化代码会查找它将类名记录为序列化格式:

class A(object):
_class_path = 'mymodule.A'
class B(object):
_class_path = 'mymodule.A.B'
...

您甚至可以使用 some metaclass magic 自动执行此操作(但也请阅读 the same SO post 中的其他评论,了解在执行上述 D=B.C 时可能适用的注意事项)。

就是说,如果您可以将序列化代码限制为 (1) 个实例新式类,以及 (2) 这些类定义在模块的顶层,那么你可以只复制 pickle 所做的(来自 Python 的 pickle.py 中第 730--768 行的函数 save_global2.6).

想法是每个新式类都定义属性__name____module__,它们是扩展为类名的字符串(如在源代码中找到)和模块名称(如在sys.modules);通过保存这些你可以稍后导入模块和获取类的实例:

__import__(module_name)
class_obj = getattr(sys.modules[module_name], class_name)

关于python - 获取python中类的类路径或 namespace ,即使它是嵌套的,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4468253/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com