在Python中递归地转换字典

5

这里是一个字典:

data = {
    'a': {
        'b': {
            'c': {
                'd': {
                    'e': {
                        'f': 1,
                        'g': 50,
                        'h': [1, 2, 4],
                        'i': 3,
                        'j': [7, 9, 6],
                        'k': [
                            [('x', 'abc')],
                            [('y', 'qwe')],
                            [('z', 'zxc')]
                        ]
                    }
                }
            }
        }
    }
}

我的目标是查找并将值转换为字典(如果可能):

data = {
    'a': {
        'b': {
            'c': {
                'd': {
                    'e': {
                        'f': 1,
                        'g': 50,
                        'h': [1, 2, 4],
                        'i': 3,
                        'j': [7, 9, 6],
                        'k': [{
                            'x': 'abc'
                        }, {
                            'y': 'qwe'
                        }, {
                            'z': 'zxc'
                        }]
                    }
                }
            }
        }
    }
}

我认为可以使用递归来实现,我甚至写了一个,但是它不起作用。

def f(d):
  for key, value in d.iteritems():
    if type(d[key]) is dict:
      f(d)

    try:
      d[key] = dict(d[key])
    except:
      if type(d[key]) is list:
        for i in d[key]:
          try:
            d[key][i] = dict(d[key][i])
          except:
            pass

  return d

错误信息:

RecursionError: 调用 Python 对象时超出了最大递归深度

如何解决该错误?
如果您能提供一个不需要递归的解决方案,我也将不胜感激。

1
你的递归在第4行:f(d) 一遍又一遍地调用相同的函数,而参数没有任何变化。这将导致堆栈溢出。我想你的意思是 f(d[key]),这样它就可以在子字典上执行 f()。 (并不是争论这是一个离题/糟糕的问题) - user955340
@DanilSperansky:这个问题缺乏明确的问题陈述;添加了错误消息。在那之前,这个问题是不相关的。如果你想讨论帖子关闭的工作原理,可以在[元]上这样做。 - Martijn Pieters
1
@DanilSperansky: 我应该多快“投票关闭”?始终立即投票关闭 - Martijn Pieters
2个回答

3

你的程序中有很多错误,让我们一起找出来,并制定一个可行的版本。

def f(d):
  for key, value in d.iteritems():
    if type(d[key]) is dict:
      f(d)                        # You should call d[key] instead 
    try:
      d[key] = dict(d[key])       # Never assign an object back to the one you are iterating over, create a new object instead.
    except:
      if type(d[key]) is list:
        for i in d[key]:
          try:
            d[key][i] = dict(d[key][i])  # This doesn't work, can't convert a tuple/list this way.
          except:
            pass

  return d

这是您的代码的修正版本,包含两个递归函数。一个用于列表,另一个用于字典。
def f1(value):
  e = []
  for val in value:
    if type(val) is list:
      e += f1(val)               # Append list to current list
    elif type(val) is tuple:
      e.append({val[0]: val[1]}) # Convert tuple to dictionary
    else:
      e.append(val)              # Append normal list values normally
  return e

def f(d, e  = {}):
  for key, value in d.iteritems():
    if type(value) is dict:
      e[key] = f(value, {})     # Recurse for dictionaries
    elif type(value) is list:
      e[key] = f1(value)        # Call the other recursive function for list
    else:
      e[key] = value            # Otherwise like strings and ints just append
  return e

Test it here: https://repl.it/LDKn/0


0

如果它是一个字典,就在d上递归调用函数,而不是value

只需将其更改为:

if type(value) is dict:
    f(value)

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接