17

我正在使用 ConfigParser 从配置文件中加载数据,如下所示:

测试.conf:

[myfiles]
fileone: %(datadir)s/somefile.foo
filetwo: %(datadir)s/nudderfile.foo

加载.py:

import ConfigParser

config = ConfigParser.ConfigParser({'datadir': '/tmp'})
config.read('test.conf')

print config.items('myfiles')
print config.get('myfiles', 'datadir')

输出:

$ python load.py 
[('datadir', '/tmp'), ('filetwo', '/tmp/nudderfile.foo'), ('fileone', '/tmp/somefile.foo')]
/tmp

我很惊讶变量替换的默认值('datadir', '/tmp')显示为 . items().get()返回,就好像它们是配置文件中的值一样。这种行为是预期的吗?任何变通方法,以便我可以简单地迭代.items()而无需在其中获取默认字典值,但仍使用魔术插值?

参考:http ://docs.python.org/library/configparser.html

谢谢!

更新:有人指出这是预期的行为:默认值就像配置文件中的任何其他名称/值对一样。同样,配置文件中的名称/值对也可用于“魔术插值”,所以如果我定义:

foo: bar
zap: %(foo)snowl

我去拿[... ('zap': 'barnowl')]

这非常简洁,但我仍然想知道我是否可以完成我想要完成的事情:迭代我的配置文件中的名称/值对,使用变量插值,没有默认值。

我的具体情况是这样的:我想用类似的东西初始化配置对象{basedir: '/foo/bar'},因为某些文件的绝对路径因安装而异。然后我需要将该配置对象传递给并让各种其他类遍历文件。我不希望每个读取配置的类都必须知道它是用某些默认值初始化的,并且应该忽略它们,因为它们不是实际文件。这可能吗?有什么方法可以隐藏 .item() 和 .get() 的默认值但仍然有插值?谢谢!

4

4 回答 4

10

一般来说,我发现configparser.Configparser类非常有用,但也缺乏。其他人也有

但是,它可以被子类化和扩展,有时很好,有时不太好(=非常依赖于实现)

这是您的问题的解决方案,在 Python3 中进行了测试:

class ConfigParser(configparser.ConfigParser):
    """Can get options() without defaults
    """
    def options(self, section, no_defaults=False, **kwargs):
        if no_defaults:
            try:
                return list(self._sections[section].keys())
            except KeyError:
                raise NoSectionError(section)
        else:
            return super().options(section, **kwargs)

这是不好的例子之一,因为它部分复制options(). 如果 configparser 基类RawConfigParser提供一个内部的选项_options(self, section)获取器,它会包含异常转换,并且options()会利用它,那就更好了。然后,在子类化中我们可以重用_options().

对于 Python 2,我相信唯一的变化是super()调用super(ConfigParser,self).

然后你可以使用:

print config.options('myfiles', no_defaults=True)

并且还使用该列表进行迭代。

于 2012-09-26T10:51:55.510 回答
4

你不能只过滤掉默认值吗?

例如:

filtered_items = [x for x in config.items('myfiles') if x[0] not in config.defaults()]

于 2016-06-07T10:51:49.383 回答
1

尝试这个:

config = ConfigParser.ConfigParser({'blahblahblah': 'blah'})
config.read('test.conf')

blahblahblah键也将出现在 中items,不是因为它是 .ini 文件中的模板,而是因为您将其指定为默认值。这就是 ConfigParser 处理默认值的方式:如果在文件中找不到它们,它会分配它们的默认值。

所以在我看来,你在这里有一个简单的概念混淆。

于 2010-02-03T04:03:14.617 回答
0

以下应该只为您提供myfiles部分中的键/值对,而不列出DEFAULT中的键/值对:

list(set(config.items('myfiles'))-set(config.items('DEFAULT')))
于 2019-01-13T21:13:51.923 回答