码迷,mamicode.com
首页 > 其他好文 > 详细

1482 路线统计

时间:2016-08-05 07:43:04      阅读:211      评论:0      收藏:0      [点我收藏+]

标签:

1482 路线统计

 

 时间限制: 1 s
 空间限制: 256000 KB
 题目等级 : 钻石 Diamond
 
 
题目描述 Description

N个节点的有向图, 求从start到finish刚好经过时间time的总方案数 mod 502630.

输入描述 Input Description

第一行包含一个整数n, 所有点是从0到N-1编号.

       接下来n行,每行包含n个字符. 第i行第j个字符表示i到j需要的时间. 字符只可能是’1’到’5’, 或者是’.’表示i不能到达j. 保证主对角线都是’.’.

       接下来一行3个整数start, finish, time.

输出描述 Output Description

输出总方案数.

样例输入 Sample Input

       3

       .12

       2.1

       12.

       0 2 5

样例输出 Sample Output

8

数据范围及提示 Data Size & Hint

       对于20%的数据, 输入的字符不是’1’就是’.’;

       对于100%的数据, 1 <= n <= 10; 1 <= start,finish <= n; 1 <= time <= 10^9.

分类标签 Tags 点此展开 

 

【解题报告】

第一眼,dfs,但看t的范围,显然超时。

再看,点很少(矩阵的n次幂耗时少),时间很多(走法复杂)

矩阵乘法的标志啊!!!

   

我们知道无边权图从s经k步到f怎么求:即01矩阵:

建立矩阵A  当且仅当存在一条边i->j ,A(i,j)=1。

令C=A*A,那么C(i,j)=ΣA(i,k)*A(k,j),实际上就等于从点i到点j恰好经过2条边的路径数(枚举k为中转点)

类似地,C*A的第i行第j列就表示从i到j经过3条边的路径数。

同理,如果要求经过k步的路径数,我们只需要二分求出A^k即可。

 

但图有边权,怎么办?

两个字:拆点!

将每个点之间的关系用矩阵存储,i能1步到j标记为1,不能到标记为0,注意题中边权为1-5,则可拆点,将每个点拆成边权个点,如图:

技术分享

 

但这样还不够,我们要建(n*n*5)^2 = 500*500的矩阵,矩阵乘法t达到500 ^ 3 这显然太多了。

 

于是:我们遇到一个边i,j,权为c,把它拆成i –> i+n*1 -> i+n*2 ->… -> i+n*(c-1)-> j

如图:

 

技术分享

于是就只有(n*5)^2=50*50的矩阵了。

 

拆完点,矩阵就变成了01矩阵

 

则这个矩中的A[i][j]就保存了1步能从i到j的方案数,要求t步,则直接将矩阵自乘t次即得答案。

 

AC代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int mod=502630;
int f,t,s,n;
char c;
int ans;
int cnt;
struct node{
    int f[60][60];
}E,x;
void clean(){
    for(int i=0;i<=n;i++) E.f[i][i]=1;
}
node cheng(node a,node b){    //矩阵乘法
    node ne;
    for(int i=0;i<n;i++){
        for(int j=0;j<n;j++){
            ne.f[i][j]=0;
            for(int k=0;k<n;k++) ne.f[i][j]=(ne.f[i][j]+((long long)a.f[i][k]*b.f[k][j])%mod)%mod;
        }
    }
    return ne;
}
int answer(){
    node ne=x;
    node ass=E;
    int b=t;
    while(b){      //矩阵求幂
        if(b&1) ass=cheng(ass,ne);
        b>>=1;
        ne=cheng(ne,ne);
    }
    return ass.f[s][f];
}
int main(){
    scanf("%d\n",&n);
    cnt=n;
    for(int i=0;i<n;i++){
        for(int j=0;j<n;j++){
            scanf("%c",&c);
            if(c>=0&&c<=9){    
                cnt=c-0;
                for(int k=1;k<cnt;k++) x.f[n*(k-1)+i][n*k+i]=1; //拆点
                x.f[n*(cnt-1)+i][j]=1;
            }
        }
        scanf("\n");
    }
    n*=5;                       //最后n要乘5
    scanf("%d%d%d",&s,&f,&t);
    clean();                    //定义单位矩阵
    printf("%d\n",answer());
    return 0;
}

 

1482 路线统计

标签:

原文地址:http://www.cnblogs.com/shenben/p/5739805.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!