亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

通過鍵列表訪問嵌套字典項?

通過鍵列表訪問嵌套字典項?

冉冉說 2019-06-12 21:35:48
通過鍵列表訪問嵌套字典項?我有一個復雜的字典結構,我想通過一個鍵列表來訪問它,以找到正確的項。dataDict = {     "a":{         "r": 1,         "s": 2,         "t": 3         },     "b":{         "u": 1,         "v": {             "x": 1,             "y": 2,             "z": 3         },         "w": 3         }}    maplist = ["a", "r"]或maplist = ["b", "v", "y"]我已經做了下面的代碼,但我相信如果有人有想法的話,有一種更好、更有效的方法來做到這一點。# Get a given data from a dictionary with position provided as a listdef getFromDict(dataDict, mapList):         for k in mapList: dataDict = dataDict[k]     return dataDict# Set a given data in a dictionary with position provided as a listdef setInDict(dataDict, mapList, value):      for k in mapList[:-1]: dataDict = dataDict[k]     dataDict[mapList[-1]] = value
查看完整描述

3 回答

?
江戶川亂折騰

TA貢獻1851條經驗 獲得超5個贊

使用reduce()翻閱字典:

from functools import reduce  # forward compatibility for Python 3import operatordef getFromDict(dataDict, mapList):
    return reduce(operator.getitem, mapList, dataDict)

再利用getFromDict若要查找要存儲值的位置,請執行以下操作setInDict():

def setInDict(dataDict, mapList, value):
    getFromDict(dataDict, mapList[:-1])[mapList[-1]] = value

除了最后一個元素mapList需要找到要添加值的“父”字典,然后使用最后一個元素將值設置為正確的鍵。

演示:

>>> getFromDict(dataDict, ["a", "r"])1>>> getFromDict(dataDict, ["b", "v", "y"])2>>> setInDict(dataDict, ["b", "v", "w"], 4)>>> 
import pprint>>> pprint.pprint(dataDict){'a': {'r': 1, 's': 2, 't': 3},
 'b': {'u': 1, 'v': {'w': 4, 'x': 1, 'y': 2, 'z': 3}, 'w': 3}}

注意,PythonPEP 8樣式指南為函數規定了Snake_case名稱..上面的方法同樣適用于列表或字典和列表的混合,所以名稱應該是get_by_path()set_by_path():

from functools import reduce  # forward compatibility for Python 3import operatordef get_by_path(root, items):
    """Access a nested object in root by item sequence."""
    return reduce(operator.getitem, items, root)def set_by_path(root, items, value):
    """Set a value in a nested object in root by item sequence."""
    get_by_path(root, items[:-1])[items[-1]] = value


查看完整回答
反對 回復 2019-06-12
?
四季花海

TA貢獻1811條經驗 獲得超5個贊

  1. 接受的解決方案不會直接對python 3起作用-它將需要一個

    from functools import reduce.

  2. 而且,使用一個

    for

    循環。見引用自

    Python3.0的新特性是什么?.

    移除reduce()..使用functools.reduce()如果您真的需要它,但是99%的時間是顯式的for循環更易讀。

  3. 接下來,接受的解決方案不會設置不存在的嵌套鍵(它返回

    KeyError

    )-見@EAFIT的答案

那么,為什么不使用Kolergy問題中建議的方法來獲得一個值:

def getFromDict(dataDict, mapList):    
    for k in mapList: dataDict = dataDict[k]
    return dataDict

以及@EAFIT用于設置值的答案中的代碼:

def nested_set(dic, keys, value):
    for key in keys[:-1]:
        dic = dic.setdefault(key, {})
    dic[keys[-1]] = value

兩者都直接在python 2和3中工作。


查看完整回答
反對 回復 2019-06-12
?
一只萌萌小番薯

TA貢獻1795條經驗 獲得超7個贊

使用REPLE很聰明,但是如果嵌套字典中不存在父鍵,OP的SET方法可能會出現問題。因為這是我在Google搜索中看到的第一篇關于這個主題的帖子,我想讓它稍微好一點。

中的集合方法在嵌套python字典中設置一個值,給出索引和值的列表)似乎對丟失的父母鑰匙更有信心。復制它:

def nested_set(dic, keys, value):
    for key in keys[:-1]:
        dic = dic.setdefault(key, {})
    dic[keys[-1]] = value

此外,有一個遍歷密鑰樹并獲取所有絕對密鑰路徑的方法也很方便,我已經為這些路徑創建了如下內容:

def keysInDict(dataDict, parent=[]):
    if not isinstance(dataDict, dict):
        return [tuple(parent)]
    else:
        return reduce(list.__add__, 
            [keysInDict(v,parent+[k]) for k,v in dataDict.items()], [])

它的一種用途是使用以下代碼將嵌套樹轉換為熊貓DataFrame(假設嵌套字典中的所有LEAF具有相同的深度)。

def dict_to_df(dataDict):
    ret = []
    for k in keysInDict(dataDict):
        v = np.array( getFromDict(dataDict, k), )
        v = pd.DataFrame(v)
        v.columns = pd.MultiIndex.from_product(list(k) + [v.columns])
        ret.append(v)
    return reduce(pd.DataFrame.join, ret)


查看完整回答
反對 回復 2019-06-12
  • 3 回答
  • 0 關注
  • 767 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號