59

我在将 YAML 文件中的文档映射到 adict并正确映射它们时遇到以下问题。

我有以下 YAML 文件,它代表服务器(db.yml):

instanceId: i-aaaaaaaa
     environment:us-east
     serverId:someServer
     awsHostname:ip-someip
     serverName:somewebsite.com
     ipAddr:192.168.0.1
     roles:[webserver,php]

我加载了这个 YAML 文件,我可以毫无问题地做到这一点,我想我明白了。

instanceId = getInstanceId()
stream = file('db.yml', 'r')
dict = yaml.load_all(stream)

for key in dict:
    if key in dict == "instanceId":
        print key, dict[key]

我希望逻辑如下所示:

  • 加载 yaml,映射到 dict
  • 查看文档中的每个 dict,如果instanceId与 设置的匹配getInstanceId(),则打印出该文档的所有键和值。

如果我从命令行查看地图数据结构,我会得到:

{'instanceId': 'i-aaaaaaaa environment:us-east serverId:someServer awsHostname:ip-someip serverName:someserver ipAddr:192.168.0.1 roles:[webserver,php]'}

我想我可能不正确地为 YAML 文件创建数据结构,并且在匹配 上的内容时dict,我有点迷失了。

旁注:我无法使用yaml.load(), 我尝试加载此文件中的所有文档yaml.load_all(),这似乎可行,但我的主要问题仍然存在。

4

6 回答 6

55

我认为你的 yaml 文件应该看起来像(或者至少是类似的,所以它的结构是正确的):

instance:
     Id: i-aaaaaaaa
     environment: us-east
     serverId: someServer
     awsHostname: ip-someip
     serverName: somewebsite.com
     ipAddr: 192.168.0.1
     roles: [webserver,php]

然后,yaml.load(...)返回:

{'instance': {'environment': 'us-east', 'roles': ['webserver', 'php'], 'awsHostname': 'ip-someip', 'serverName': 'somewebsite.com', 'ipAddr': '192.168.0.1', 'serverId': 'someServer', 'Id': 'i-aaaaaaaa'}}

你可以从那里去...


所以像这样使用:

>>> for key, value in yaml.load(open('test.txt'))['instance'].iteritems():
    print key, value


environment us-east
roles ['webserver', 'php']
awsHostname ip-someip
serverName somewebsite.com
ipAddr 192.168.0.1
serverId someServer
Id i-aaaaaaaa
于 2012-10-22T21:40:30.037 回答
7

您的代码中的另一个错误,与 YAML 无关:

for key in dict:
    if key in dict == "instanceId": # This doesn't do what you want
        print key, dict[key]

in是一个适用于序列类型和地图的运算符。这就是为什么这不是语法错误......但它没有做你想要的。

key in dict将始终评估为True,因为您正在迭代的所有键都在字典中。因此,您的代码归结为True == "instanceId",它将始终评估为False,因为布尔值True永远不会等于该字符串。

您可能已经注意到该print语句不会产生任何输出。这是因为它永远不会被调用。

于 2017-03-07T16:35:01.657 回答
5

只需使用python-benedict,它是一个 dict 子类,为大多数常见格式提供 I/O 支持,包括yaml.

from benedict import benedict

# path can be a yaml string, a filepath or a remote url
path = 'path/to/data.yml'

d = benedict.from_yaml(path)

# do stuff with your dict
# ...

# write it back to disk
d.to_yaml(filepath=path)

它经过了很好的测试和记录,请查看 README 以查看所有功能: https ://github.com/fabiocaccamo/python-benedict

使用 pip 安装:pip install python-benedict

注意:我是这个项目的作者

于 2019-09-20T16:40:13.600 回答
3

您可以使用biosPython3 的包,如下所示:

import bios

my_dict = bios.read('data.yml')

bios正在从文件中读取原始数据并转换 python dict 对象。根据文件的扩展名,可以判断出文件的类型。

于 2019-10-02T13:46:07.363 回答
2

我喜欢用 Path,做一个漂亮的单线

import yaml
from pathlib import Path
conf = yaml.safe_load(Path('data.yml').read_text())
于 2021-12-26T18:18:21.797 回答
1

如果您遇到了这个问题,试图弄清楚如何使用 pyyaml 库从 yaml 文件中获取 python 字典,请尝试safe_load如下所示的选项。

import yaml
yaml_dict = yaml.safe_load("data.yml")
于 2021-12-17T13:19:17.140 回答