矩阵链应用中括号的可能组合

Aly*_*ono 8 python algorithm math matrix python-3.x

我研究了矩阵链乘法,其中给定一系列矩阵,目标是找到最有效的乘法矩阵方法.问题实际上不是执行乘法,而只是决定所涉及的矩阵乘法的顺序.这就是为什么我的任务是创建一个程序,在矩阵乘法中输出所有可能的矩阵组合,给定n作为矩阵的数量作为输入.例如

 n == 1     (A)

 n == 2     (AB)

 n == 3     (AB)C ,  A(BC)

 n== 4      ((AB)C)D,   (A(BC))D, A((BC)D), A(B(CD)), (AB)(CD)
Run Code Online (Sandbox Code Playgroud)

我的初始代码在下面,被称为

 possible_groupings(4) #4 matrices

def possible_groupings(n):
    print("Possible Groupings : ")
    total  = 0
    if(n==1):
        print('A')
        total = total + 1
    elif(n==2):
       print('(AB)')
       total = total + 1
    else:
       a = 2
       while(a <= n-1):
           b = 0
           while((b+a) <= (n )):
               c = b

               d = 0
               substr = ''
               while (d < c):                    
                   substr = substr + chr(65 + d)                    
                   d = d + 1

               if substr != '':
                   if len(substr) == 1:
                      print( substr, end = '')
                   else:
                      print('(' + substr + ')', end = '')

            print('(', end = '')
            while (c < (b +a)):                    
                print(chr(65 + c), end = '');
                c = c + 1
            print(')', end = '')

            e = b+a

            substr = ''
            while (e < n):
                substr = substr + chr(65 + e) 
                e = e + 1
            if substr != '':
                if len(substr) == 1:
                    print( substr, end = '')
                else:
                    print('(' + substr + ')', end = '')
            print('')

            total = total + 1

            b = b + 1
        a = a + 1
print('Total : ' + str(total))
Run Code Online (Sandbox Code Playgroud)

当我的inout是4个矩阵时,上面代码的输出是:

(AB)(CD)
A(BC)D
(AB)(CD)
(ABC)D
A(BCD)
Run Code Online (Sandbox Code Playgroud)

我该如何修改我的代码.矩阵的数量必须在1-26范围内.我的脑袋现在疼痛了.请帮忙.

Pau*_*zer 6

这是一个从前到后工作的递归方案.

它作为生成器实现part,以最后一次乘法开始.最后的乘法必须在两个因子之间,其中左边是第一个j(cut下面的代码中的变量)矩阵("左边块")的产品,右边是剩余矩阵的产品("右边块") ).j可以是1到N -1 之间的任何值,其中N是链中矩阵的数量.

因此,为了枚举所有分组,我们必须遍历j.对于每个j,我们必须将左块的每个分组与右块的每个分组组合.要枚举我们part自己使用的块的分组,即递归.

def part(names, top=True):
    lr = ('', '') if top else '()'
    if len(names) <= 1:
        yield names
    elif len(names)==2:
        yield names.join(lr)
    else:
        for cut in range(1, len(names)):
            for left in part(names[:cut], False):
                for right in part(names[cut:], False):
                    yield (left+right).join(lr)
Run Code Online (Sandbox Code Playgroud)

最小化器可以使用相同的逻辑.这可以利用以下提供的memoization functools.lru_cache:

from functools import lru_cache
from string import ascii_uppercase

@lru_cache(None)
def _min_no_mult(dims):
    if len(dims) == 2:
        return 0, 'x'
    elif len(dims)==3:
        return dims[0]*dims[1]*dims[2], 'xx'.join('()')
    cuts = ((cut, *_min_no_mult(dims[:cut+1]), *_min_no_mult(dims[cut:]))
            for cut in range(1, len(dims)-1))
    return min((mnl + mnr + dims[0]*dims[-1]*dims[cut], (nml+nmr).join('()'))
                for cut, mnl, nml, mnr, nmr in cuts)

def min_no_mult(dims, names=None):
    mn, argmn = _min_no_mult(tuple(dims))
    names = iter(ascii_uppercase if names is None else names)
    argmn = argmn[1:-1] if len(dims) > 2 else argmn
    argmn = ''.join(next(names) if a=='x' else a for a in argmn)
    return mn, argmn
Run Code Online (Sandbox Code Playgroud)

演示:

>>> for i, j in enumerate(part(ascii_uppercase[:6])):
...     print(i, j)
... 
0 A(B(C(D(EF))))
1 A(B(C((DE)F)))
2 A(B((CD)(EF)))
3 A(B((C(DE))F))
4 A(B(((CD)E)F))

...

38 ((A((BC)D))E)F
39 (((AB)(CD))E)F
40 (((A(BC))D)E)F
41 ((((AB)C)D)E)F
Run Code Online (Sandbox Code Playgroud)

由于记忆,最小化器可以轻松处理大量维度:

>>> import numpy as np
>>> dims = np.clip(np.arange(-1, 26), 1, None)
>>> np.random.shuffle(dims)
>>> dims
array([ 5, 25,  1,  4, 14, 24,  7, 15,  2, 12, 11,  9, 18,  8, 19, 13, 23,
       17,  1, 22, 21,  1, 16,  6,  3, 20, 10])

>>> min_no_mult(dims)
(3383, '(AB)((((((((((CD)E)F)G)H)(I(J(K(L(M(N(O(P(QR))))))))))((ST)U))((VW)X))Y)Z)')
Run Code Online (Sandbox Code Playgroud)

我们可以查询一些基本的缓存统计:

>>> _min_no_mult.cache_info()
CacheInfo(hits=5450, misses=351, maxsize=None, currsize=351)
Run Code Online (Sandbox Code Playgroud)

这可能看起来不那么令人印象深刻,但请记住,每次点击都会削减整个子树.

实际上,我们可以再次回收重复计划并计算括号数:

@lru_cache(None)
def count(n):
    if n <= 2:
        return 1
    else:
        return sum(count(cut) * count(n-cut) for cut in range(1, n))
Run Code Online (Sandbox Code Playgroud)

对于26个矩阵,有很多方法可以将它们括起来:

>>> print(f"{count(26):,d}")
4,861,946,401,452
Run Code Online (Sandbox Code Playgroud)


mka*_*kam 3

看起来您想要将字符集划分为所有可能的子集,尽管您似乎没有考虑不连续的分组(例如(AC)(DB))。如果是这样,那么这就是一个众所周知的问题,并且存在众所周知的解决方案。例如,请参阅如何查找集合的所有分区