『题目』:
你是一位施工队的工长,根据设计师的要求准备为一套设计风格独特的房子进行室内装修。
房子的客厅大小为 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块(但其实这个是有问题的,因为没有枚举小块的大小
和切块的位置
):
这里还有一个可视化图片,最正规的做法应该是暴力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));
}
}