『Leetcode 5241』铺瓷砖

『题目』:

 你是一位施工队的工长,根据设计师的要求准备为一套设计风格独特的房子进行室内装修。

房子的客厅大小为 n x m,为保持极简的风格,需要使用尽可能少的 正方形 瓷砖来铺盖地面。

假设正方形瓷砖的规格不限,边长都是整数。

请你帮设计师计算一下,最少需要用到多少块方形瓷砖?

『限制条件』:

1 <= n <= 13
1 <= m <= 13

『输入输出』

输入:n = 2, m = 3
输出:3
解释:3 块地砖就可以铺满卧室。
     2 块 1x1 地砖
     1 块 2x2 地砖
     
输入:n = 5, m = 8
输出:5

输入:n = 11, m = 13
输出:6

『题解』:

这道题可以用dp来做,就是dp[i][j]代表切分长为i和宽为j时候的最小块数,其中需要分三种情况dp(i==j时,dp[i][j]=1):

  • 竖着切割,将矩形分成左右两块。
  • 横着切割,将矩阵分成上下两块。
  • i*j大小的矩阵中,挖一个1*1的小块,然后将矩阵分成5块(但其实这个是有问题的,因为没有枚举小块的大小切块的位置):
    『Leetcode 5241』铺瓷砖_第1张图片
    这里还有一个可视化图片,最正规的做法应该是暴力dfs,每次查看哪个一位置空了,然后填写矩阵下去,意思是暴力模拟才更准确些。

『实现』:

public class test4 {
    
    int tilingRectangle(int n, int m) {

        int [][] dp = new int[15][15];

        for(int i = 1;i <= n;i++)
        {
            for(int j = 1;j<= m;j++)
            {
                if(i == j)
                {
                    dp[i][j] = 1;
                    continue;
                }
                else dp[i][j] = i * j;

                for(int sli = 1; sli < i;sli ++)
                    dp[i][j] = Math.min(dp[i][j],dp[sli][j] + dp[i - sli][j]);
                for(int sli = 1; sli < j;sli ++)
                    dp[i][j] = Math.min(dp[i][j],dp[i][sli] + dp[i][j - sli]);
                /*
                    移动一个1*1大小的块,然后将矩阵分成4块,形成5块
                 */
                for(int x = 2; x < i;x++)
                    for(int y = 2;y < j; y++)
                    {
                        dp[i][j]=Math.min(dp[i][j], dp[x-1][y]+dp[x][j-y]+dp[i-x+1][y-1]+dp[i-x][j-y+1]+1);
                    }
            }
        }

        return dp[n][m];
    }

    public static void main(String[] args) {

        test4 of = new test4();
        System.out.println(of.tilingRectangle(11,13));
    }
}

你可能感兴趣的:(『数据结构与算法』)