zoukankan      html  css  js  c++  java
  • CodeForces 166E Tetrahedron解题报告

    这是本人写的第一次博客,学了半年的基础C语言,初学算法,若有错误还请指正。

                                                          E. Tetrahedron

                                                                              time limit per test2 seconds
                                                                         memory limit per test256 megabytes
                                                                                    inputstandard input
                                                                                  outputstandard output
     You are given a tetrahedron. Let's mark its vertices with letters A, B, C and D correspondingly.
                                                                           

    An ant is standing in the vertex D of the tetrahedron. The ant is quite active and he wouldn't stay idle. At each moment of time he makes a step from one vertex to another one along some edge of the tetrahedron. The ant just can't stand on one place.

    You do not have to do much to solve the problem: your task is to count the number of ways in which the ant can go from the initial vertex D to itself in exactly n steps. In other words, you are asked to find out the number of different cyclic paths with the length of n from vertex D to itself. As the number can be quite large, you should print it modulo 1000000007 (109 + 7).

    Input
    The first line contains the only integer n (1 ≤ n ≤ 107) — the required length of the cyclic path.

    Output
    Print the only integer — the required number of ways modulo 1000000007 (109 + 7).

    Sample test(s)
    input
    2
    output
    3
    input
    4
    output
    21
    Note
    The required paths in the first sample are:

    D - A - D
    D - B - D
    D - C - D
     
     

    笔者高中的时候记得在数学试卷上做过类似的题目, 不过当时是求概率的,所以笔者一看到这道题目的时候就是往概率上想,而走n步的情况总可能就是3的n次方,所以只要求出走n步到达D点的概率,用它乘以总可能的情况数并对10的9次方加7取余便是最终的答案,这里设P(A, n)为第n次到A点的概率,所以P(A, n) + P(B, n) + P(C, n) + P(D, n) = 1; 且由于对称性可以知道,P(A, n)  = P(B, n) = P(C, n), 即P(A, n) = 1/3 * (1 - P(D, n))(1),而第n次到达D点,说明第n - 1次在A, B, C 中的其中一点,而又由于对称性可得,P(D, n) = 1/3*(P(A, n - 1) + P(B, n - 1) + P(C, n - 1)) = P(A, n - 1); 与(1)式联立得P(D, n + 1) = 1/3 * (1 - P(D, n)); 利用高中数列知识可以求得;则第n次到达D点的方法数等于总数(3 ^ n)乘以概率,设第n次到达D点的方法数为dp[n], 则有
    ;这个可以直接用暴力写,也可以用矩阵快速幂写,在这就不贴代码了,不过高中学过数学竞赛的同学可能会觉得这个式子结构()有点眼熟, 3和-1的次数都是等次幂,让人联想到线性递推数列的求解 : 对于数列An = p * An-1 +  q *An-2, 其特征方程为x^2 = px + q, 假设其两根分别为x1, x2,则(x1, x2不相等时)或(x1 = x2时),a,b为常数,可由给出条件获得,所以可以把中的3和-1分别看成x1,x2,由二次方程(设为x^2 + px + q = 0)的韦达定理的p = -(x1 + x2) = -2, q = x1 *x2 = 3;即方程为x^2 - 2x - 3 = 0,而此方程为数列An = 2 * An-1 +  3 *An-2
    的特征方程,由此我们可以得到一个递推关系式为dp[n] = 2 * dp[n - 1] + 3 *dp[n - 2];
    下面附上代码:
     1 #include <cstdio>
     2 #include <cstring>
     3 const int N = 1e7 + 5;
     4 const int mod = 1e9 + 7;
     5 long long dp[N], n;
     6 void init(){
     7     dp[1] = 0;
     8     dp[2] = 3;
     9     for(int i = 3; i < N; i ++){
    10         dp[i] = 2 * dp[i - 1] + 3 * dp[i - 2];
    11         dp[i] %= mod;
    12     }
    13 }
    14 int main(){
    15     init();
    16     while(scanf("%d", &n) == 1)
    17         printf("%d\n", dp[n]);
    18     return 0;
    19 }

    
    
    
    不过刚开始这点思路非常复杂,所有笔者就去想dp[n] = 2 * dp[n - 1] + 3 *dp[n - 2]的含义
    在纠结了好些时辰之后,终于想通了,我们可以第n - 1步到达D点的情况中把最后一步移到其他2个点,再加一步到D点(假如说本来第n - 1步到达D点的情况为A ->D, 那么先让A走到B或C点再走到D点而此时的补数即为n), 而如果还是要走D点那样必须再多走2步才能再次回到D点,即由本来A ->D变成了A->D->X->D。
    X可能为A或B或C,即有三种可能的情况,所以也是为什么dp[n - 2]还要乘以3了。
    其实我们没有必要开这么大的数列对a, b每次做不同的更新就行了,就是时间会比较费:
     1 #include <cstdio>
     2 #include <cstring>
     3 #include <algorithm>
     4 using namespace std;
     5 const int mod = 1e9 + 7;
     6 int n;
     7 long long a, b;
     8 int main(){
     9     while(scanf("%d", &n) == 1){
    10         if(n < 2)
    11             printf("0\n");
    12         else if(n == 2)
    13             printf("3\n");
    14         else{
    15             a = 0;
    16             b = 3;
    17             for(int i = 1; i <= n - 2; i ++){
    18                 if(i & 1)
    19                     a = (3 * a + 2 * b)%mod;
    20                 else
    21                     b = (3 * b + 2 * a)%mod;
    22             }
    23             if(n & 1)
    24                 printf("%I64d\n", a);
    25             else
    26                 printf("%I64d\n", b);
    27         }
    28     }
    29     return 0;
    30 }
    
    
    还有一种思路:假设dp[i][j] 表示第i步走到j点(用0表示D点,1,2,3分别表示A,B,C三个点)则有转移方程:
    dp[i][j] += dp[i - 1][k] (k, j 不相等),初始化为dp[1][1] = dp[1][2] = dp[1][3] = 1;
     1 #include <cstdio>
     2 #include <cstring>
     3 typedef long long lld;
     4 const int N = 1e7 + 5;
     5 const int mod = 1e9 + 7;
     6 int dp[N][5], n;
     7 void init(){
     8     dp[1][1] = 1;
     9     dp[1][2] = 1;
    10     dp[1][3] = 1;
    11     for(int i = 2; i < N; i ++){
    12         for(int j = 0; j < 4; j ++){
    13             for(int k = 0; k < 4; k ++){
    14                 if(k == j )
    15                         continue;
    16                 dp[i][j] += dp[i - 1][k];
    17                 dp[i][j] %= mod;
    18             }
    19         }
    20     }
    21 }
    22 int main(){
    23     init();
    24     while(scanf("%d", &n) == 1)
    25         printf("%d\n", dp[n][0]);
    26     return 0;
    27 }

    
    
    
    其实可以从这个递推关系式可以推出dp[n] = 2 * dp[n - 1] + 3 *dp[n - 2]
    解:dp[n][0] = dp[n - 1][1] + dp[n - 1][2] + dp[n - 1][3] = 3 * dp[n - 1][1](A,B,C三点具有对称性)
    则dp[n - 1][1] = 1/3 * dp[n][0];
           dp[n - 1][1] = 2 * dp[n - 2][1] + dp[n - 2][0];
    联立以上两式,则可得dp[n][0] = 2 * dp[n - 1][0] + 3 *dp[n - 2][0];
    即为dp[n] = 2 * dp[n - 1] + 3 *dp[n - 2];
     
    其实再次观察,便可得出另一个关系式:
    dp[n] = 3 * dp[n - 1] - 3(n为奇数时)
    3 * dp[n - 1] +3(n为偶数时)
     1 #include <cstdio>
     2 #include <cstring>
     3 const int N = 1e7 + 5;
     4 const int mod = 1e9 + 7;
     5 long long dp[N], n;
     6 void init(){
     7     dp[1] = 0;
     8     dp[2] = 3;
     9     for(int i = 3; i < N; i ++){
    10         if(i & 1)
    11             dp[i] = 3 * dp[i - 1] - 3;
    12         else
    13             dp[i] = 3 * dp[i - 1] + 3;
    14         dp[i] %= mod;
    15     }
    16 }
    17 int main(){
    18     init();
    19     while(scanf("%d", &n) == 1)
    20         printf("%d\n", dp[n]);
    21     return 0;
    22 }
    既然要做,那就好好做! 自己选的路,自己走完!
  • 相关阅读:
    windows下读取utf-8文件
    mongodb c api编译
    JS学习笔记
    【已解决】关于SQL2008 “不允许保存更改。您所做的更改要求删除并重新创建以下表。您对无法重新创建的标进行了更改或者启用了‘阻止保存要求重新创建表的更改’” 解决方案
    C#字符串截取
    C#通过WatiN操作页面中内嵌的Iframe
    C#查找以某个字母开头另一字母结尾的字符串
    HTML5+CSS3学习小记
    C#求任意范围内的质数
    C#中的委托
  • 原文地址:https://www.cnblogs.com/zyf0163/p/4318036.html
Copyright © 2011-2022 走看看