• BZOJ 1297: [SCOI2009]迷路 [矩阵快速幂]


    Description

    windy在有向图中迷路了。 该有向图有 N 个节点,windy从节点 0 出发,他必须恰好在 T 时刻到达节点 N-1。 现在给出该有向图,你能告诉windy总共有多少种不同的路径吗? 注意:windy不能在某个节点逗留,且通过某有向边的时间严格为给定的时间。

    Input

    第一行包含两个整数,N T。 接下来有 N 行,每行一个长度为 N 的字符串。 第i行第j列为'0'表示从节点i到节点j没有边。 为'1'到'9'表示从节点i到节点j需要耗费的时间。

    Output

    包含一个整数,可能的路径数,这个数可能很大,只需输出这个数除以2009的余数。

    Sample Input

    【输入样例一】
    2 2
    11
    00

    【输入样例二】
    5 30
    12045
    07105
    47805
    12024
    12345


    Sample Output

    【输出样例一】
    1

    【样例解释一】
    0->0->1

    【输出样例二】
    852

    HINT

    30%的数据,满足 2 <= N <= 5 ; 1 <= T <= 30 。
    100%的数据,满足 2 <= N <= 10 ; 1 <= T <= 1000000000 。

    思路:矩阵快速幂应该是第一个能想到的,但是直接将一个长为9的边拆成9个点,那最坏情况下就有9*9*9个点约等于700多个点,时间复杂度是n*n*n*log(t)前面显然会爆,但是可以这样,把一个点拆成9个点,9个点连成一条链,这样就可以乱搞了,如果一个点x到这个点y有长度为k的边 只要将x连到y前面k-1个点就行(因为连出一条边就减少了一条边)

     1 #include<cstdio>
     2 #include<iostream>
     3 #include<cstring>
     4 #define N 90
     5 #define MOD 2009
     6 using namespace std;
     7 char ch[100][100];
     8 struct mat
     9 {
    10     long long m[N+1][N+1];
    11     mat(){memset(m,0,sizeof(m));}
    12 };
    13 mat operator *(mat a,mat b)
    14 {
    15     mat ans;
    16     for(int i=1;i<=N;i++)
    17     {
    18         for(int j=1;j<=N;j++)
    19         {
    20             for(int k=1;k<=N;k++)
    21             {
    22                 ans.m[i][j] = (ans.m[i][j] + a.m[i][k] * b.m[k][j])% MOD;
    23             }
    24         }
    25     }
    26     return ans;
    27 }
    28 mat pow(mat a,long long n)
    29 {
    30     mat ret;
    31     for(int i=1;i<=N;i++)ret.m[i][i]=1;
    32     for(;n;n>>=1)
    33     {
    34         if(n&1)ret = (ret * a);
    35         a = (a*a);
    36     }
    37     return ret;
    38 }
    39 int main()
    40 {
    41     int n,t;
    42     mat a;
    43     scanf("%d%d",&n,&t);
    44     for(int i=1;i<=n;i++)
    45     {
    46         scanf("%s",ch[i]+1);
    47     }
    48     for(int i=1;i<=n;i++)
    49     {
    50         for(int j=1;j<=8;j++)
    51         {
    52             a.m[(i-1)*9+j][(i-1)*9+j+1]=1;
    53         }
    54     }
    55     for(int i=1;i<=n;i++)
    56     {
    57         for(int j=1;j<=n;j++)
    58         {
    59             int u = ch[i][j]-'0';
    60             if(u!=0)
    61             {
    62                 a.m[(i-1)*9+9][(j-1)*9+(9-u+1)]=1;
    63             }
    64         }
    65     }
    66     a = pow(a,t);
    67     printf("%lld
    ",a.m[9][(n-1)*9+9]);
    68     return 0;
    69 }
  • 相关阅读:
    IE6/IE7浏览器中"float: right"自动换行的解决方法
    IE6/IE7浏览器不支持display: inline-block;的解决方法
    如何解决两个li之间的缝隙
    input、button、a标签 等定义的按钮尺寸的兼容性问题
    在一个页面重复使用一个js函数的方法
    关于让input=text,checkbox居中的解决方法
    遮盖层实现(jQuery+css+html)
    button,input type=button按钮在IE和w3c,firefox浏览器区别
    前端-选项卡(菜单栏)
    形成人、机器、过程和数据的互联互通
  • 原文地址:https://www.cnblogs.com/philippica/p/4700637.html
Copyright © 2020-2023  润新知