Skip to content Skip to sidebar Skip to footer

Calculating The Complexity Of Algorithm To Print All Valid (i.e., Properly Opened And Closed) Combinations Of N-pairs Of Parentheses

I would like your opinion on the time and space complexity of this algorithm I implemented (in Python) to calculate the complexity of algorithm to print all valid (i.e., properly o

Solution 1:

For optimal results, avoid sets. If you generate each possibility exactly once, you can just put them into a vector.

Here's a very simple recursion which is slightly slower than @bcdan's answer:

def rget(n):
  ifn== 0:
    return ['']
  else:
    return [fst + '(' + snd + ')'for i in range(n)for fst in rget(i)for snd in rget(n-i-1)]

If you wanted a generator rather than a complete list of results, a variant of the above is probably ideal, but for the case of a complete list, it's quite a bit faster to compute the results of each j from 0 to n, using the previously computed lists instead of a recursive call:

def iget(n):
  res = [['']]
  for j in range(1, n+1):
    res.append([fst + '(' + snd + ')'for i in range(j)for fst in rget(i)for snd in rget(j-i-1)])
  return res[n]

This turns out to be an order of magnitude faster (using Python 3.3.2).

Why it works

You can decompose any balanced string into two balanced substrings. This is not a unique decomposition, but it can be made unique by choosing the shortest possible non-empty balanced suffix. The shortest nonempty balanced suffix has the property that the starting ( matches the closing ); if that were not the case, it could be decomposed into two shorter non-empty balanced sequences. So the recursive consists of finding all sequences of the form Fst(Snd) where the sum of the sizes of Fst and Snd is n-1.

Time complexity:

The number of balanced sequences with n pairs of parentheses is the nCatalan number Cn, which is O(4n). The above code generates each sequence in O(n) (because the string concatenation takes O(n) time) for a total complexity of O(4n)

Solution 2:

Let's try with a simpler version instead.

Some theorems:

  1. Properly paired string are of even length. Don't ask me to prove this one, please.
  2. Given a properly paired string of length 2k, it is possible to construct all strings of length 2(k + 1) by inserting pairs of parenthesis '()' at every possible location within the string. There are 2k + 1 such positions.
  3. To generate all n pairs, we need to recursive into the insertion step n times (and get strings of length 2n.

Note that this is not enough to generate all unique properly paired strings, because e.g. inserting () into () yields the same string twice (()()). However as an upper bound it should suffice.

def add_parens(s, nparens):    
    if not s:    
       add_parens('()', nparens)    
    max_length = 2 * nparens    
       if len(s) == max_length:    
       print(s)    
    else:    
        for i in range(0, len(s)):    
            add_parens(s[0:i] + '()' + s[i:], nparens)    

n = 5add_parens('', n)  

Time complexity:

  1. There is 1 insertion point for the empty string.
  2. There are 3 insertion points for (). ...

So all in all:

T(n) = 1 * 3 * ... * 2n + 1 ~ O(n!)

Space complexity for the recursive version is O(n(2n + 1)), however I'm pretty sure it is possible to bring this down to linear.

Solution 3:

Recursive version seems to be very simple and it spends time only on valid branches:

defgen(sofar, l, r, n):
    """
    sofar: string generated so far
    l: used left parentheses
    r: used right parentheses
    n: total required pairs
    """
    result = []
    if l == n and r == n:
        # solution found
        result.append(sofar)
    else:
        if l > r:
            # can close
            result.extend(gen(sofar + ")", l, r + 1, n))
        if l < n:
            # can open
            result.extend(gen(sofar + "(", l + 1, r, n))
    return result

defgenerate(n):
    return gen("", 0, 0, n)

print generate(4)

Solution 4:

The number of possible combinations is the Catalan Number. So the complexity is at least the one specified by that number. https://en.wikipedia.org/wiki/Catalan_number

Solution 5:

I got curious and wrote my own version. Here are some specs (this is all in Python 2.7):

n=8

Mine: 21 ms

Yours: 89 ms

n=10

Mine: 294 ms

Yours: 1564 ms

def get(n):
    
    def find(current):
        out= []
        last_index =0for j inrange(current.count('(')):
            last_index = current.index('(',last_index) +1
            out.append(current[:last_index] +'()'+current[last_index:])
        returnout
        
    seed ='()'current='()'
    temp =set(['()'])
    for i inrange(n):
        new=set()
        for thing in temp:
            new.update(find(thing))
        temp =newreturn [a[1:-1] for a in temp]

I think the biggest difference is that mine only has 3 for loops, and yours has 4. This happens at the str.count function. Using this built-in probably contributes significantly to speed. Also, after each stage I identified duplicates, and this cuts down exponentially on time.

Actually, the biggest difference I see is that I only insert after parenthesis, and then strip the outer two when it is finished. It creates way fewer duplicates. So ((())()) is created, and reduced to its correct form of (())() after removing the enclosing ().

Post a Comment for "Calculating The Complexity Of Algorithm To Print All Valid (i.e., Properly Opened And Closed) Combinations Of N-pairs Of Parentheses"