Я следующий список: -Python - Создание словаря (дерево) из списка кортежей
a = [(1, 1), (2, 1), (3, 1), (4, 3), (5, 3), (6, 3), (7, 7), (8, 7), (9, 7)]
который представляет собой список кортежей. Элементы внутри кортежа имеют формат (Id, ParentId)
Его корневой узел всякий раз, когда Id == ParentId
. Список может быть в любом порядке кортежей.
Я хочу создать следующий словарь, используя приведенный выше список кортежей,
output = [{
'id': 1,
'children': [{
{
'id': 3,
'children': [{
{
'id': 5
},
{
'id': 4
},
{
'id': 6
}
}]
},
{
'id': 2
}
}]
}, {
'id': 7,
'children': [{
{
'id': 9
},
{
'id': 8
}
}]
}]
т.е. (в терминах graphs- в Форрест)
1 7
/\ /\
2 3 8 9
/|\
4 5 6
Мой окончательный вывод должен быть словарь дал выше.
Я попробовал следующее: -
Решение, которое я попробовал следующий: -
# set the value of nested dictionary.
def set_nested(d, path, value):
reduce(lambda d, k: d.setdefault(k, {}), path[:-1], d)[path[-1]] = value
return d
# returns the path of any node in list format
def return_parent(list, child):
for tuple in list:
id, parent_id = tuple
if parent_id == id == child:
return [parent_id]
elif id == child:
return [child] + return_parent(list, parent_id)
paths = []
temp = {}
for i in a:
id, parent_id = i
temp[id] = {'id': id}
path = return_parent(a, id)[::-1]
paths.append(path) # List of path is created
d = {}
for path in paths:
for n, id in enumerate(path):
set_nested(d, path[:n + 1], temp[id]) # setting the value of nested dictionary.
print d
Выход, который я получил,
{
'1': {
'3': {
'6': {
'id': '6'
},
'5': {
'id': '5'
},
'id': '3',
'4': {
'10': {
'id': '10'
},
'id': '4'
}
},
'2': {
'id': '2'
},
'id': '1'
},
'7': {
'9': {
'id': '9'
},
'8': {
'id': '8'
},
'id': '7'
}
}
Я близко к нему, но не в состоянии получить точный результат. Кроме того, есть ли лучшее лучшее решение?
просто интересно, но вы гарантированно иметь действительный вход? Нет циклов (например, 1 является родительским для 2, который является родительским для 1)? –