NYOJ 980 格子刷油漆 动态规划

这道题目状态转移方程比较复杂,刚开始以为没这么多情况,看了好多大牛的博客再加上与同学讨论才看懂,写下心得。

因为起点不固定,所以我们一个一个来考虑,先从角上考虑,设三个数组来表示分别为D,A,Sum,分别表示为“从一个角开始然后回到同一列的对应位置的总个数”, “从一个角开始的总个数(包括回到对应位置和不回到对应位置)”, “表示总的个数”

1. 当回到对应位置时: 

  D[1] = 1,D[n] = 2 * D[n-1];

  因为他可以有两种方式出去,最后再回来,如图(a), 图上是以右上角这个格子出发的,然后回到右下角这个格子,一共有两种方式出去。

2. 不一定回到对应位置时,就是一个角的总个数:

  A[n] = D[n] + 2*A[n - 1] + 4 * A[n - 2];

  其中, D[n]表示回到对应位置的,而A[n-1]这种情况如图(b)所示,先上对面去,然后再接着向前走,所以还是两种方式出去,后面这种表示挨着走完两列,就是前面没有的这种情况,一共有两种走法,但是每种走法又有两种出去的方式,所以是4种,如下图(c)表示的,黑色(1,2两种)的和红色的(3,4两种)箭头方向。

  这只是其中一个角,因为只要n>2就有4个角,所以这个所有角的个数就是4*A[n];

3.当从中间位置开始走时:

  假设从 i 开始走, 2*(2*D[i-1]*2*A[n-i] + 2*D[n-i]*2*A[i-1]),最后从2 - n-1 遍历一遍加起来

  假设从 i 开始走,显然不能直接往下走,否则无法遍历所有点,应当是先遍历左边(右边)所有点,然后回到相对的点,然后遍历右边(左边)的点。注意先遍历的时候,必须是采用“遍历全体格子后回到与之相对的格子”的走法,否则无法遍历出发点正下方的点,而后遍历则不受限制。所以先往左走的 i 之前的方法总数就是D[i - 1], 之后A[n - i],同理可以推先往右走的。

附AC代码:

 1 #include <iostream>
 2 #include <cstdio>
 3 using namespace std;
 4 const int N = 1005;
 5 const long long mod = 1000000007;
 6 long long D[N], A[N], Sum[N];//D数组表示从一个角出发,最后回到出发点的同一列所对应的那个格子的总个数,
 7 //A数组表示所有一个角出发的总个数,Sum表示总的个数
 8 int main()
 9 {
10     D[1] = 1;
11     A[1] = 1;
12     A[2] = 6;//初始条件,从一个角开始所有的总数
13     Sum[1] = 2;//总数
14     Sum[2] = 24;
15     for (int i = 2; i < N; i++)
16         D[i] = D[i - 1] * 2 % mod;
17     for (int i = 3; i < N; i++)
18         A[i] = (D[i] + 2 * A[i - 1] + 4 * A[i - 2]) % mod;
19     for (int i = 3; i < N; i++)
20     {
21         Sum[i] = 4 * A[i] % mod;
22         for (int k = 2; k < i; k++)
23             Sum[i] = (Sum[i] + 8 * D[k - 1] * A[i - k] % mod + 8 * D[i - k] * A[k - 1] % mod) % mod;
24     }
25     int n;
26     while (cin >> n)
27     {
28         cout << Sum[n] << endl;
29     }
30 
31     return 0;
32 }
View Code

 

posted @ 2015-03-23 21:46  Howe_Young  阅读(329)  评论(0编辑  收藏  举报