import java.util.Scanner; public class Asist { /** * (1) * dp1[i][j] 代表 (i+2) * (j+2)个方块组成的格子进行 1 轮的总方法数 !(数学归纳法获知的!——可以打印出来看看) * * (2) * 关于这个式子:sum=sum+test(k-1, i-2, j-2)*(n-i+1)*(m-j+1); * a) test(k, i, j) 求的是 i * j 个方块组成的格子进行k轮的总可行数! (废话了!) * * b) 至于这个式子的来历,我举个实际例子吧,(一般我的解题思路都是先举出一个实际例子、然后对其他情况加以证明而完成的、) * * 1. 对于给定的n, m, k, 必须满足 n >= 2k+1 && m >= 2k+1才可有可行方案,否则输出 0; (很容易证明, 也是关键点!) * * 例子: * n m k * 7 8 3 * 说明: 每进行一轮时, 此时方框的最外层是不可取的, 也就是说,实际最大可取方框为 (n-2) * (m-2) * * 第一轮, 应至少取出 x * y 个方框,(其中 n >= x >= 2*k - 1 && m >= y >= 2*k - 1 ),此时 n = 5, m = 6, k = 3, * 这样做方面配合我的 dp1[][]数组,才这样的。 * 这一轮的可行取法有: * i,j * 1. 5,5 这种取法的方式有 2 次 —— * 次数 = (n-i+1) * (m-j+1) (很容易证明) * 2. 5,6 这种取法的方式有 1次 —— * * 第二轮, 根据前一轮的答案,同理的做法。 第一轮中有两种符合的情况: * 1. n m k * 5 5 2 * 完成一轮操作, 应至少取出 x * y 个方框,(其中 n >= x >= 2*k - 1 && m >= y >= 2*k - 1 ),此时 n = 3, m = 3, k = 2, * 这种情况,这一轮的取法有: * i, j * 1. 3, 3 这种取法的方式有1次 * 2. n m k * 5 6 2 * 完成一轮操作, 应至少取出 x * y 个方框,(其中 n >= x >= 2*k - 1 && m >= y >= 2*k - 1 ),此时 n = 3, m = 4, k = 2, * 这种情况,这一轮的取法有: * i, j * 1. 3, 3 这种取法的方式有2次 * 2. 3, 4 这种取法的方式有1次 * 第三轮,根据前一轮的答案,同理 * 1. n m k * 3 3 1 * 当 k = 1时,就好说了, 因为我的dp1[] 就是解决 k=1的。 此时n = 1, m = 1, k = 1, * 结果是: dp[1][1] * 1(第三轮1次) * 1(第二轮1次) * 2(第一轮2次) = 2 * 2. n m k * 3 3 1 * 结果是: dp[1][1] * 1(第三轮1次) * 2(第二轮2次) * 1(第一轮1次) = 2 * 3. n m k * 3 4 1 * 结果是: dp[1][2] * 1 * 1 * 1 = 3; * * 综上所述: sum = 2 + 2 + 3 = 7; */ public static final int MOD = 1000000007; public static int dp1[][]; // 记录一个初始状态 public static long dp2[][] = new long[1000][1000]; // 记忆化搜索 public static void main(String[] args) { Scanner sc = new Scanner(System.in); int n = sc.nextInt(); int m = sc.nextInt(); int k = sc.nextInt(); // 初始化 dp1 数组 dp1 = new int[1000][1000]; for (int i = 1; i < 1000 ; i++) { dp1[1][i] = i * (i+1) / 2; dp1[i][1] = dp1[1][i]; } for (int i = 2; i < 1000; i++) { for (int j = 2; j < 1000; j++) { dp1[i][j] = dp1[i][1] * dp1[1][j]; } } //模拟游戏 System.out.println(test(k, n-2, m-2)); } private static long test(int k, int n, int m) { if (k == 1) { return dp1[n][m]; } if (dp2[n][m]!=0) return dp2[n][m]; long sum = 0; int temp = 2*k-1; for (int i = temp; i <= n; i++) { for (int j = temp; j <= m; j++) { sum = (sum + (test(k-1, i-2, j-2)%MOD * (n-i+1)%MOD * (m-j+1)%MOD) % MOD)%MOD; } } dp2[n][m] = sum; // 记忆 return sum; } }
import java.util.Scanner; public class Main { public static final int MOD = 1000000007; public static void main(String[] args) { Scanner sc = new Scanner(System.in); int n = sc.nextInt(); int m = sc.nextInt(); int k = sc.nextInt(); System.out.println(test(2*k,n-1)*test(2*k,m-1)%MOD); } private static long test(int n, int m) { if (n > (m / 2)) n = m-n; long sum = 1; for (int i = 0; i < n; i++) sum = (sum * (m-i)) % MOD; long sun = 1; for (int i = 2; i <= n; i++) sun = (sun * i) % MOD; return sum / sun; } }
原文地址:http://blog.csdn.net/first_sight/article/details/45724543