Size: 4907
Comment: added problems 14, 15
|
Size: 4917
Comment:
|
Deletions are marked like this. | Additions are marked like this. |
Line 118: | Line 118: |
Line 128: | Line 129: |
= Problem 14: Duplicate the elements of a list | = Problem 14: Duplicate the elements of a list = |
Line 134: | Line 136: |
= Problem 15: Duplicate the elements of a list a given number of times | = Problem 15: Duplicate the elements of a list a given number of times = |
Python Solutions to [https://prof.ti.bfh.ch/hew1/informatik3/prolog/p-99/ 99 Prolog Prolems].
Index TableOfContents(2)
Problems 1-6
André Roberge has a zip file with solutions to the first six problems, in Crunchy format: [http://crunchy.googlecode.com/files/first_6_of_99_problems.zip First six]
Problem 7: Flatten a nested list structure
Based on the standard library documentation:
from itertools import chain def flatten(listOfLists): return list(chain(*listOfLists))
The suggested solution does not work for a list like the following:
a_list = [0, 1, [2, 3], 4, 5, [6, 7]]
as the argument name tries to imply, it only works for a list of lists, not a generic list of variously and mixedly nested lists and items. Here's a more general solution using the simple recursive approach:
def flatten(nestedList): def aux(listOrItem): if isinstance(listOrItem, list): for elem in listOrItem: for item in aux(elem): yield item else: yield listOrItem return list(aux(nestedList))
This problem is also a good example of "recursion elimination": explicitly maintain a LIFO stack of what sublists are being expanded so as to avoid actual recursion. The rec-elim approach is usually faster and avoids issues with recursion depth limits. Here's a version that works when it's OK to dismantle the input argument -- for variety, I have it build the result into another list by calls to .append, instead of using yield in an auxliary generator and calling list() on it.
def flatten(nestedList): result = [] stack = [nestedList] while stack: if isinstance(stack[-1], list): try: stack.append(stack[-1].pop(0)) except IndexError: stack.pop() # remove now-empty sublist else: result.append(stack.pop()) return result
If you're not allowed to dismantle the input argument, you can take a preliminary copy.deepcopy of it as the initial item in the stack, or you can "pay as you go" by doing shallow copies "at the last minute" when needed. Here's an example of the latter approach, with other little variants. Here, stack is always a list of non-empty sublists which are shallow copies of sublists from the initial argument (and so the sublists on the stack can always be dismantled with no problems) while leaves (non-list subitems) are always immediately appended to the result (this, btw, builds up the result in a reversed way, so a call to result.reverse becomes necessary).
def flatten(nestedList): result = [] if not nestedList: return result stack = [list(nestedList)] while stack: current = stack.pop() next = current.pop() if current: stack.append(current) if isinstance(next, list): if next: stack.append(list(next)) else: result.append(next) result.reverse() return result
Problem 8: Eliminate consecutive duplicates of list elements
from itertools import groupby def compress(alist): return [key for key, group in groupby(alist)]
Problem 9: Pack consecutive duplicates of list elements into sublists
from itertools import groupby def pack(alist): return [list(group) for key, group in groupby(alist)]
Problem 10: Run-length encoding of a list
from itertools import groupby def encode(alist): return [[len(list(group)), key] for key, group in groupby(alist)]
Problem 11: Modified run-length encoding
def encode_modified(alist): def aux(lg): if len(lg)>1: return [len(lg), lg[0]] else: return lg[0] return [aux(list(group)) for key, group in groupby(alist)]
Problem 12: Decode a run-length encoded list
def decode(alist): def aux(g): if isinstance(g, list): return [(g[1], range(g[0]))] else: return [(g, [0])] return [x for g in alist for x, R in aux(g) for i in R]
Problem 13: Run-length encoding of a list (direct solution)
def encode_direct(alist): def aux(k, g): l = sum(1 for x in g) if l>1: return [l, k] else: return k return [aux(key, group) for key, group in groupby(alist)]
Problem 14: Duplicate the elements of a list
def dupli(L): return [x for x in L for i in (1,2)]
Problem 15: Duplicate the elements of a list a given number of times
def dupli(L, N): return [x for x in L for i in range(N)]