标签:技术分享 计数 while 输入数据 ogr input col height 相加
一道计数问题,重点就是要做到不重不漏.
怎样步数最短呢?当然是每个格子都只经过1次啦.
然后我们注意到如果这个起点选在中间的第i列,我们就把这个格子划分成了左右两个部分
感觉就是对于某个起点然后左右两部分是之前求过的答案直接相乘,不同起点再相加就可以了,但是关键问题是我们怎样计数呢?
我们设 Bn 代表从某个角(如左上角)出发,然后走遍所有格子回到同一列的方案数目。
同样,我们设 An 代表从某个角出发,然后走遍所有格子的方案数。
下面我们来解释这个式子
第一项表示从某个角开始最后回到起点的同一列的那个角
第二项表示从某个角开始,第二步走与起点相同列的那个角,第三步我们向第二列走的时候就有两种走法了(选择直接向右或者走对角线,乘的那个2)
从第四步开始我们现在面对的问题就是了
第三项中4可以看成2*2,
我们把前两列写成这个样子
1 | 2 |
3 | 4 |
第一个2代表走一个(1 2 3 4)和(1 4 3 2)有两种走法起点在第一列而终点在第二列
第二个2代表对于每一个到达第二列后第五步向第三列走的时候都有两种走法(向右或者走对角线)
从第六步开始时我们面对的问题就是了
我们现在来考虑下正经问题
首先有4个角,那么ans先加上
但是如果我们起点选在中间呢?
假设我们选在了第i列,首先这一列有两个格子 所以先乘个2
现在我们可以选择先往左跑,我们可以直接向左,也可以走一个向左的对角线.再乘个2
注意往左跑完1~i-1列以后一点要跑到第i列那个"不是起点的点"否则没法往右跑,所以乘的是
现在我们又回到了第i列,现在我们向右跑,又有两种情况,直接向右,向右的对角线 再乘个2
现在我们到了i+1列对于右半部分(i+1~n)我们只要跑完就行了,我们不关心终点所以乘
这样一开始向左跑就是,一开始向右跑自然是
1 #include <bits/stdc++.h> 2 3 using namespace std; 4 typedef long long ll; 5 const int maxn = 11000; 6 const ll mod = 1e9+7; 7 ll a[maxn],b[maxn]; 8 void init() 9 { 10 b[1]=1; 11 for (int i=2;i<maxn;++i){ 12 b[i] = (b[i-1]*2)%mod; 13 } 14 a[1] = 1; 15 a[2] = 6; 16 for (int i=3;i<maxn;++i) 17 a[i] = (b[i]%mod+(2*a[i-1]%mod)+(4*a[i-2])%mod)%mod; 18 } 19 int n; 20 int main() 21 { 22 init(); 23 int T; 24 scanf("%d",&T); 25 while (T--){ 26 scanf("%d",&n); 27 if (n==1){ 28 printf("%d\n",2); 29 } 30 else{ 31 ll ans = 0; 32 ans = (a[n]*4)%mod; 33 for (int i=2;i<n;++i){ 34 ans = (ans+2*( ( ( (2*b[i-1])%mod) *( (2*a[n-i])%mod ) + ( (2*a[i-1])%mod) *( (2*b[n-i])%mod ) )%mod )%mod)%mod; 35 } 36 printf("%lld\n",ans); 37 } 38 } 39 return 0; 40 }
标签:技术分享 计数 while 输入数据 ogr input col height 相加
原文地址:http://www.cnblogs.com/agenthtb/p/7516302.html