1

我正在使用带有端点原型数据存储的 protopc。

我想从提供的结构构建自定义消息。

例如,它是以下键列表:['id1', 'id2', 'id3']

每个键都分配给一个MessageField命名的CustomField.

我想继承Message一个包含所有key.

def create_custom_container(key_list):
    class cls():
        pass
    for i, k in enumerate(key_list):
        setattr(cls, k, MessageField(CustomField, i))
    return cls

class CustomMessage(Message, create_custom_container(key_list)):
    pass

但它不起作用,我得到: MessageDefinitionError: Message types may only inherit from Message

我从protorpc 源代码中看到Message使用元类来防止它被继承或动态修改属性。

所以,我不知道如何即时创建我的自定义消息。

4

1 回答 1

3

该库在为 Message 类定义约束方面走了很长一段路 - 破解它以强制新属性可能会导致 Message 根本无法按预期工作。

幸运的是,不是使用 class CustomMessage语句对类主体进行硬编码,而是使用调用创建自定义类 - 它允许您以编程方式定义内容。这样您就不需要在继承树上使用多个类。

您所要做的就是Message使用适当的参数调用 's 元类,而不是通常调用type,并作为类命名空间传递 -

所以你可以重写你的身体创造功能:

def create_custom_body(key_list):
    dct = {}
    for i, k in enumerate(key_list):
        dct[k] = MessageField(CustomField, i)
    return dct

CustomClass  = Message.__class__("CustomClass", (Message,), create_custom_body(key_list))

这将在这种情况下起作用。但是,如果库的元类将使用自定义命名空间(即它会有一个 __prepare__方法),则需要修改它以使用types.new_class和适当的回调:

from types import new_class

def create_custom_body(dct, key_list):
    for i, k in enumerate(key_list):
        dct[k] = MessageField(CustomField, i)
    return dct

CustomClass  = types.new_class(
    "CustomClass", (Message,), 
    exec_body=(lambda namespace: create_custom_body(namespace, key_list))
)

(查看文档:https ://docs.python.org/3/library/types.html )

于 2018-03-06T13:15:49.343 回答