I'm trying to solve a problem that, unfortunately, goes beyond my capacity. I have a series of nested lists and while iterating over them, in case the next element is a list I want to append it as an attribute of my current element. As usual, an example is better than my poor English (here's some code to copy and paste):
class T(object):
def __init__(self, id, children):
self.id = id
self.children = children or []
def __repr__(self):
return u"T(id={0}, children={1})".format(self.id, self.children)
# first a short example
l0 = [T(id=1, children=[]), T(id=2, children=[]), [T(id=3, children=[]),
T(id=4, children=[]), [T(id=5, children=[])]]]
As you can see l0 has 3 elements and the last one is list of three elements: what I need is to append the last list to the previous element that is not a list (and recursively) Expected output:
l1 = magic(l0)
[T(id=1, children=[]), T(id=2, children=[T(id=3, children=[]), T(id=4, children=[T(id=5, children=[])])])]
Hope somebody can share some advice to solve this, I've already invested a lot of hours and I'm not even close to solve it.
EDIT
For completeness, here's a little more complex example
l0 = [T(children=[], id=1),
T(children=[], id=2),
T(children=[], id=3),
[T(children=[], id=40),
T(children=[], id=41),
T(children=[], id=42),
T(children=[], id=43),
T(children=[], id=44),
T(children=[], id=45),
[T(children=[], id=50),
T(children=[], id=51),
T(children=[], id=52),
T(children=[], id=54),
[T(children=[], id=60),
T(children=[], id=61),
T(children=[], id=62),
T(children=[], id=63),
[T(children=[], id=70)],
T(children=[], id=64)]]],
T(children=[], id=8),
T(children=[], id=9)]
I built a doctest using @rik-poggi function as example and so far it seems to be ok:
>>> from magic_bag import magic
>>> class T(object):
... def __init__(self, id, children):
... self.id = id
... self.children = children or []
...
... def __repr__(self):
... return u"T(children={0}, id={1})".format(self.children, self.id)
...
>>> l0 = [T(id=1, children=[]), T(id=2, children=[]), T(id=3, children=[]),
... [T(id=40, children=[]), T(id=41, children=[]), T(id=42, children=[]),
... T(id=43, children=[]), T(id=44, children=[]), T(id=45, children=[]),
... [T(id=50, children=[]), T(id=51, children=[]), T(id=52, children=[]),
... T(id=54, children=[]), [T(id=60, children=[]), T(id=61, children=[]),
... T(id=62, children=[]), T(id=63, children=[])]]], T(id=8, children=[]),
... T(id=9, children=[])]
>>> l1 = magic(l0)
>>> l1[0]
T(children=[], id=1)
>>> l1[1]
T(children=[], id=2)
>>> l1[3]
T(children=[], id=8)
>>> l1[4]
T(children=[], id=9)
>>> l1[5]
Traceback (most recent call last):
...
IndexError: list index out of range
>>> l1[2].children[5].children[3]
T(children=[T(children=[], id=60), T(children=[], id=61), T(children=[], id=62), T(children=[], id=63)], id=54)
>>> l0 = [T(id=1, children=[]), T(id=2, children=[]), T(id=3, children=[]),
... [T(id=40, children=[]), T(id=41, children=[]), T(id=42, children=[]),
... T(id=43, children=[]), T(id=44, children=[]), T(id=45, children=[]),
... [T(id=50, children=[]), T(id=51, children=[]), T(id=52, children=[]),
... T(id=54, children=[]), [T(id=60, children=[]), T(id=61, children=[]),
... T(id=62, children=[]), T(id=63, children=[]), [T(id=70, children=[])],
... T(id=64, children=[])]]], T(id=8, children=[]), T(id=9, children=[])]
>>> l1 = magic(l0)
>>> l1[2].children[5].children[0].id
50
>>> len(l1[2].children[5].children[3].children)
5
>>> l1[2].children[5].children[3].children[3].children
[T(children=[], id=70)]
>>> l1[2].children[5].children[3].children[4].id==64
True
Using @rob-wouters alternative, it passes the same test too so for the test cases I tried both work ok. I will keep Rik's because I think a standalone function can be more handy for the cases when I need this kind of behavior.
[[1] [2]]or[[1] 2]or[1 [2] [3]]or[1 [2] [3] 4]or[1 [2 [3]]]