0756-金字塔转换矩阵

Raphael Liu Lv10

你正在把积木堆成金字塔。每个块都有一个颜色,用一个字母表示。每一行的块比它下面的行 少一个块 ,并且居中。

为了使金字塔美观,只有特定的 三角形图案 是允许的。一个三角形的图案由 两个块 和叠在上面的 单个块
组成。模式是以三个字母字符串的列表形式 allowed 给出的,其中模式的前两个字符分别表示左右底部块,第三个字符表示顶部块。

  • 例如,"ABC" 表示一个三角形图案,其中一个 “C” 块堆叠在一个 'A' 块(左)和一个 'B' 块(右)之上。请注意,这与 "BAC" 不同,"B" 在左下角,"A" 在右下角。

你从底部的一排积木 bottom 开始,作为一个单一的字符串,你 必须 使用作为金字塔的底部。

在给定 bottomallowed 的情况下,如果你能一直构建到金字塔顶部,使金字塔中的 每个三角形图案 都是允许的,则返回
true ,否则返回 false

示例 1:

**输入:** bottom = "BCD", allowed = ["BCC","CDE","CEA","FFF"]
**输出:** true
**解释:** 允许的三角形模式显示在右边。
从最底层(第3层)开始,我们可以在第2层构建“CE”,然后在第1层构建“E”。
金字塔中有三种三角形图案,分别是“BCC”、“CDE”和“CEA”。都是允许的。

示例 2:

**输入:** bottom = "AAAA", allowed = ["AAB","AAC","BCD","BBE","DEF"]
**输出:** false
**解释:** 允许的三角形模式显示在右边。
从最底层(游戏邦注:即第4个关卡)开始,创造第3个关卡有多种方法,但如果尝试所有可能性,你便会在创造第1个关卡前陷入困境。

提示:

  • 2 <= bottom.length <= 6
  • 0 <= allowed.length <= 216
  • allowed[i].length == 3
  • 所有输入字符串中的字母来自集合 {'A', 'B', 'C', 'D', 'E', 'F', 'G'}
  • allowed 中所有值都是 唯一的

方法一:状态转换

算法:

我们模拟方块可以处于的状态。每个状态都是一个二进制数,如果第 k 类型的方块是可能的,则设置第 k 位。然后,我们创建一个转换映射 T[state1][state2] -> state。它接受左状态和右状态并输出所有可能的父状态。

最后,应用这些转换非常简单。但是,这种方法不正确,因为转换不是独立的。例如,如果我们在一行 A, {B or C}, A,并且 allowed 中的元组是 (A, B, D), (C, A, D)。那么无论选择 {B or C},我们都不能创建金字塔的下一行。

[solution1-Python]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class Solution(object):
def pyramidTransition(self, bottom, allowed):
T = [[0] * (1 << 7) for _ in xrange(1 << 7)]
for triple in allowed:
u, v, w = (1 << (ord(x) - ord('A')) for x in triple)
for b1 in xrange(1 << 7):
if u & b1:
for b2 in xrange(1 << 7):
if v & b2:
T[b1][b2] |= w

state = [1 << (ord(x) - ord('A')) for x in bottom]
while len(state) > 1:
for i in xrange(len(state) - 1):
state[i] = T[state[i]][state[i+1]]
state.pop()
return bool(state[0])
[solution1-Java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
class Solution {
public boolean pyramidTransition(String bottom, List<String> allowed) {
int[][] T = new int[1 << 7][1 << 7];
for (String triple: allowed) {
int u = 1 << (triple.charAt(0) - 'A');
int v = 1 << (triple.charAt(1) - 'A');
int w = 1 << (triple.charAt(2) - 'A');
for (int b1 = 0; b1 < (1 << 7); ++b1) if ((u & b1) > 0)
for (int b2 = 0; b2 < (1 << 7); ++b2) if ((v & b2) > 0)
T[b1][b2] |= w;
}

int[] state = new int[bottom.length()];
int t = 0;
for (char c: bottom.toCharArray())
state[t++] = 1 << (c - 'A');
while (t-- > 1)
for (int i = 0; i < t; ++i)
state[i] = T[state[i]][state[i+1]];
return state[0] > 0;
}
}

复杂度分析

  • 时间复杂度:O(2^{2\mathcal{A}}A + N^2)。其中 N 指的是 bottom 的长度,A 指的是 allowed 的长度,且 \mathcal{A 指的是字母的大小。
  • 空间复杂度:O(2^{2\mathcal{A}})。

方法二:深度优先搜索

我们详尽的尝试每个方块的组合。

算法:

我们需要从三元组列表中创建一个转换映射 T。这个映射 T[x][y] = {set of z} 将是左孩子 x 和右孩子 y 所有可能的父块。

然后,为了求解下一行,我们生成下一行所有的可能组合并求解它们。如果这些组合中有任一一行是可解的,则返回 True,反之返回 False

[solution1-Python]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
class Solution(object):
def pyramidTransition(self, bottom, allowed):
T = collections.defaultdict(set)
for u, v, w in allowed:
T[u, v].add(w)

#Comments can be used to cache intermediate results
#seen = set()
def solve(A):
if len(A) == 1: return True
#if A in seen: return False
#seen.add(A)
return any(solve(cand) for cand in build(A, []))

def build(A, ans, i = 0):
if i + 1 == len(A):
yield "".join(ans)
else:
for w in T[A[i], A[i+1]]:
ans.append(w)
for result in build(A, ans, i+1):
yield result
ans.pop()

return solve(bottom)
[solution1-Java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
class Solution {
int[][] T;
Set<Long> seen;

public boolean pyramidTransition(String bottom, List<String> allowed) {
T = new int[7][7];
for (String a: allowed)
T[a.charAt(0) - 'A'][a.charAt(1) - 'A'] |= 1 << (a.charAt(2) - 'A');

seen = new HashSet();
int N = bottom.length();
int[][] A = new int[N][N];
int t = 0;
for (char c: bottom.toCharArray())
A[N-1][t++] = c - 'A';
return solve(A, 0, N-1, 0);
}

//A[i] - the ith row of the pyramid
//R - integer representing the current row of the pyramid
//N - length of current row we are calculating
//i - index of how far in the current row we are calculating
//Returns true iff pyramid can be built
public boolean solve(int[][] A, long R, int N, int i) {
if (N == 1 && i == 1) { // If successfully placed entire pyramid
return true;
} else if (i == N) {
if (seen.contains(R)) return false; // If we've already tried this row, give up
seen.add(R); // Add row to cache
return solve(A, 0, N-1, 0); // Calculate next row
} else {
// w's jth bit is true iff block #j could be
// a parent of A[N][i] and A[N][i+1]
int w = T[A[N][i]][A[N][i+1]];
// for each set bit in w...
for (int b = 0; b < 7; ++b) if (((w >> b) & 1) != 0) {
A[N-1][i] = b; //set parent to be equal to block #b
//If rest of pyramid can be built, return true
//R represents current row, now with ith bit set to b+1
// in base 8.
if (solve(A, R * 8 + (b+1), N, i+1)) return true;
}
return false;
}
}
}

复杂度分析

  • 时间复杂度:O(\mathcal{A}^{N}),其中 N 指的是 bottom 的长度,\mathcal{A 指的是字母的大小。
  • 空间复杂度:O(N^2)。
 Comments