https://github.com/September26/java-algorithms
有两种形状的瓷砖:一种是 2 x 1
的多米诺形,另一种是形如 "L" 的托米诺形。两种形状都可以旋转。
给定整数 n ,返回可以平铺 2 x n
的面板的方法的数量。返回对 109 + 7
取模 的值。
平铺指的是每个正方形都必须有瓷砖覆盖。两个平铺不同,当且仅当面板上有四个方向上的相邻单元中的两个,使得恰好有一个平铺有一个瓷砖占据两个正方形。
示例 1:
输入: n = 3 输出: 5 解释: 五种不同的方法如上所示。
示例 2:
输入: n = 1 输出: 1
提示:
1 <= n <= 1000
* 解题思路: * 这一定是一道动态规划的题,如果我们去掉L型,那么就是一个斐波那契额数列,F(n)=F(n-1)+F(n-2) * 加上L形状,我们仍然可以按照这个思路来。 * 开头只有四种可能,-,=,L,「。 * -和=开头的,我们可以归结为斐波那契额数列,f(n)=f(n-1)+f(n-2) * L和「开头的,我们可以归结为一类问题,使用F(n)来表示,所以可以转换为f(n)=f(n-1)+f(n-2)+F(n) * 我们再来看下F(n)怎解决? * L开头的话,只有两种可能: * L开头,以┐结尾,中间包含若干-形,其数量为f(n-3)+f(n-5)+f(n-7)... * L开头,以」结尾,中间包含若干-形状,其数量为f(n-4)+f(n-6)+f(n-8)... * 两种类型累加,就是F(n)的数量。 * 最后,每次计算对值求模,得到我们想要的结果
- public class Solution790 {
-
- Map<Integer, Integer> fMap = new HashMap<>();
- int flag = 10_0000_0000 + 7;
-
- public int numTilings(int n) {
- fMap.put(0, 1);
- fMap.put(1, 1);
- fMap.put(2, 2);
- int index = 3;
- while (index <= n) {
- int indexValue = fMap.get(index - 1) % flag + fMap.get(index - 2) % flag;
- indexValue = indexValue % flag;
- indexValue += (2 * F(index)) % flag;
- indexValue = indexValue % flag;
- fMap.put(index, indexValue);
- index++;
- }
- return fMap.get(n);
- }
-
- private int F(int index) {
- //L开头,」结尾
- int sum = 0;
- for (int i = 0; i <= index - 3; i += 2) {
- sum += fMap.get(index - 3 - i) % flag;
- sum = sum % flag;
- }
- if (index <= 3) {
- return sum;
- }
- for (int i = 0; i <= index - 4; i += 2) {
- sum += (fMap.get(index - 4 - i) % flag);
- sum = sum % flag;
- }
- //L开头,┐结尾
- return sum;
- }
- }