11

TLDR 总结

我写了一个navigateDict在 a 上进行安全导航的函数dict,类似于dict.get()但嵌套。它取代了类似的代码

if 1 in data and 'i' in data[1] and 'a' in data[1]['i']:
    print data[1]['i']['a']
else:
    print "Not found"

与大致相当

found = navigateDict(data, 1, 'i', 'a')
if found is not None:
    print found
else:
    print "Not found"
  • 有什么类似于这已经是标准库的一部分吗?
  • 有没有更惯用的方法来做同样的事情?
    • 任何需要多次键入任何路径组件键的响应都可能是非答案。

额外细节

实现如下:

# Allow fallback value other than None
def navigateDictEx(d, keys, fallback=None):
    for key in keys:
        if key in d:
            d = d[key]
        else:
            return fallback
    return d

def navigateDict(d, *keys):
    return navigateDictEx(d, keys)

有关示例用法,请参阅摘要。

不管是否是 Pythonic,这个函数在冗余是一个坏主意的地方减少了重复。例如,在示例中更改一个路径组件需要将最多三个不同的值修改为原始示例中的一个,但在修改后的示例中只需要一个。鉴于我经常犯错误的倾向,这是一个巨大的胜利。

最终我要问的是:标准库中有什么东西可以做到这一点,还是我需要在我的项目库中为它找到一个位置?

如果预计命中率会主导未命中率

brionius 正确地指出,捕捉KeyError将起作用:

try:
    print data[1]['i']['a']
except KeyError:
    print "Not found"

这可能是我要走的路;它非常简洁,减少了重复。然而,它确实反映了一个假设,即命中比未命中要多。如果有更好的假设相反的方法,我也想知道。

4

3 回答 3

10

一种方法如下:

try:
    print data[1]['i']['a']
except KeyError:
    print "Not found!"

这符合鸭式打字的精神。它可能会也可能不会那么快,因为我相信处理异常会带来一定的开销,但它肯定是“安全的”。

于 2013-08-22T16:07:30.243 回答
4

这样的解决方案很酷

https://twitter.com/raymondh/status/343823801278140417

>>> from collections import defaultdict
>>> infinite_defaultdict = lambda: defaultdict(infinite_defaultdict)
>>> d = infinite_defaultdict() 
>>> d['x']['y']['z'] = 10
>>> if d['x']['y']['z']: print d['x']['y']['z'] #better reflects that misses are common
于 2013-08-22T16:20:04.027 回答
-1

游戏晚了好几年,但对于任何偶然发现这一点的人来说,似乎仍然没有一种原生、流畅的方式来安全地导航 Python 字典。

输入RestResponse

“RestResponse 旨在成为一个流畅的 Python 对象,用于与 RESTful JSON API 交互”

这个库包括一个允许安全导航(和构建)JSON 数据结构的 NoneProp 对象。

>>> import RestResponse
>>> data = RestResponse.parse({})
>>> data.property.is_none
None
>>> bool(data.property.is_none)
False
>>> isinstance(data.property.is_none, RestResponse.NoneProp)
True
>>> data.property.is_none = None
>>> isinstance(data.property.is_none, RestResponse.NoneProp)
False
>>> print data.pretty_print()
{
    "property": {
        "is_none": null
    }
}
于 2018-02-27T15:07:32.693 回答