我正在尝试为一些配置对象定制一个YAML转储器/加载程序。为了简单起见,假设我们要将Hero
类的对象转储到hero.yml
文件中。在
class Hero:
yaml_tag = '!Hero'
def __init__(self, name, age):
self.name = name
self.age = age
然后通过ruamel.yaml
添加默认的加载程序/转储程序
试着卸载:
h = Hero('Saber', 15)
with open('config.yml', 'w') as fout:
yaml.dump(h, fout)
with open('config.yml') as fin:
yaml.load(fin)
它工作得很好!在
to_yaml
和from_yaml
方法失败但是,当我需要一个更灵活的行为时,因此需要一个自定义的from_yaml
和{
Hero
的实现更改为:
class Hero:
yaml_tag = '!Hero'
def __init__(self, name, age):
self.name = name
self.age = age
@classmethod
def to_yaml(cls, representer, data):
return representer.represent_mapping(cls.yaml_tag,
{'name': data.name, 'age': data.age})
@classmethod
def from_yaml(cls, constructor, node):
print(node) # for debug
value = constructor.construct_mapping(node)
return cls(**value)
翻斗车工作正常。但是加载未能加载YAML文件。安 引发异常:
243 def check_mapping_key(self, node, key_node, mapping, key, value):
244 # type: (Any, Any, Any, Any, Any) -> None
--> 245 if key in mapping:
246 if not self.allow_duplicate_keys:
247 args = [
TypeError: argument of type 'NoneType' is not iterable
通过标记为print(node)
的print(node)
行,加载的节点是:
MappingNode(tag='!Hero', value=[(ScalarNode(tag='tag:yaml.org,2002:str', value='name'), ScalarNode(tag='tag:yaml.org,2002:str', value='Saber')), (ScalarNode(tag='tag:yaml.org,2002:str', value='age'), ScalarNode(tag='tag:yaml.org,2002:int', value='15'))])
这个例子是一个很小的例子来说明这个问题,在真实的例子中,我试图只转储对象的一部分,比如
class A:
yaml_tag = '!A'
def __init__(self, name, age):
self.data = {'name': name, 'age': age}
所需的A('Saber', 15)
的YAML文件是
!A
name: Saber
age: 15
在这种情况下,我不知道如何使默认的卸载程序/装载程序工作。在
我的错误在哪里导致失败?如何解决这个问题?在
RoundTripConstructor.construct_mapping
的定义是:它需要知道它需要什么样的映射 构造。在旋转木马里有一些期待什么 可以附加到这样一个对象上,所以你最好模拟 RoundTripDumper过程中的例程:
CommentedMap
(一个普通的dict
不起作用)。在所以你需要做一些事情,比如:
^{pr2}$它给出了:
上面的工作原理是因为你的类相对简单,如果可以的话 递归部分,您需要遵循两步创建过程, 以创建对象的初始收益率,以便在递归过程中使用它。在
maptyp
默认为None
是历史的,它必须是 设置。E、 g.construct_mapping
首先要做的事情之一是 正在尝试附加注释(如果节点上有可用的注释)。 我将删除0.15.55中的默认值,这将使 如果你像你一样漏掉了,那就错了。在相关问题 更多 >
编程相关推荐