1297: [SCOI2009]迷路


Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 1385  Solved: 993
[Submit][Status][Discuss]

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


【输入样例一】



【输入样例二】


Sample Output


【输出样例一】



【样例解释一】

->->


【输出样例二】



HINT


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

分析:


  先用dp的思想分析可以分析出转移方程dp[i][t] = ∑dp[j][t  -  time(i,j)];但是t太大,并且时间也不允许;

  想了一下发现转移方程是固定的,可以很自然联想矩阵乘法 + 快速幂。
  但是写的时候出现困难了,发现每个点对间有些彼此到达时间各不相同不好处理。于是就困住了

正解:


 

可以把一个点拆出9个虚拟结点出来,node[i][j] 向 node[i][j + ]点向虚拟结点连边为1,然后所有到达这个点的都连边向node[i][],所有以这个点出发能到距离为j的点q从node[i][j - ]连边1向node[q][];

于是原图换成了一个10 * 10的矩阵,矩乘 + 快速幂求出即行。

复杂度还是很低的。

矩阵乘法:


其实关于dp的东西如果发现是固定转移并且是刷表什么的,都可以联想到转换成矩乘来做。

AC代码:


# include <iostream>
# include <cstdio>
# include <cstring>
using namespace std;
const int mod = ;
int n;
long long t;
struct fi{
int data[][];
fi(){memset(data,,sizeof(data));
}
}A,T;
inline fi operator * (fi & a,fi & b){
fi t;
for(int i = ;i < n * ;i++){
for(int j = ;j < n * ;j++){
t.data[i][j] = ;
for(int k = ;k < n * ;k++){
t.data[i][j] += a.data[i][k] * b.data[k][j];
}
if(t.data[i][j] >= mod)t.data[i][j] %= mod;
}
}
return t;
}
inline fi operator +(fi & a,fi & b){
fi t;
for(int i = ;i < n * ;i++){
for(int j = ;j < n * ;j++){
t.data[i][j] = a.data[i][j] + b.data[i][j];
if(t.data[i][j] >= mod)t.data[i][j] %= mod;
}
}
return t;
}
inline void cmd(fi & A,fi & T,long long k){
while(k){
if(k & 1LL)A = A * T;
T = T * T;
k >>= 1LL;
}
return;
}
char str[];
void build(){
scanf("%d %lld",&n,&t);
int x;
for(int i = ;i < n;i++){
scanf("%s",str);
for(int j = ;j < n;j++){
x = str[j] - '';
if(!x)continue;
x -= ;
T.data[i * + x][j * ] = ;
}
for(int j = ;j < ;j++){
T.data[i * + j][i * + j + ] = ;
}
}
A.data[][] = ;
cmd(A,T,t);
printf("%d\n",A.data[][(n - ) * ]);
}
int main(){
build();
}

[Bzoj1297][Scoi2009 ]迷路 (矩阵乘法 + 拆点)的更多相关文章

  1. bzoj1297: [SCOI2009]迷路(矩阵乘法+拆点)

    题目大意:有向图里10个点,点与点之间距离不超过9,问从1刚好走过T距离到达n的方案数. 当时看到这题就想到了某道奶牛题(戳我).这两道题的区别就是奶牛题问的是走T条边,这道题是每条边都有一个边权求走 ...

  2. BZOJ1297 [SCOI2009]迷路 矩阵乘法

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1297 题意概括 有向图有 N 个节点,从节点 0 出发,他必须恰好在 T 时刻到达节点 N-1. ...

  3. 【bzoj1297】[SCOI2009]迷路 矩阵乘法

    题目描述 给出一个 $n$ 个点的有向图,每条边的权值都在 $[1,9]$ 之间.给出 $t$ ,求从 $1$ 到 $n$ ,经过路径边权和恰好为 $t$ 的方案数模2009. 输入 第一行包含两个整 ...

  4. [luogu4159 SCOI2009] 迷路(矩阵乘法)

    传送门 Solution 矩阵乘法新姿势qwq 我们知道当边权为1是我们可以利用矩阵快速幂来方便的求出路径数 那么对于边权很小的时候,我们可以将每个点都拆成若干个点 然后就将边权不为1转化为边权为1了 ...

  5. LUOGU P4159 [SCOI2009]迷路(矩阵乘法)

    传送门 解题思路 以前bpw讲过的一道题,顺便复习一下矩阵乘法.做法就是拆点,把每个点拆成\(9\)个点,然后挨个连边.之后若\(i\)与\(j\)之间的边长度为\(x\),就让\(i\)的第\(x\ ...

  6. BZOJ1297: [SCOI2009]迷路 矩阵快速幂

    Description windy在有向图中迷路了. 该有向图有 N 个节点,windy从节点 0 出发,他必须恰好在 T 时刻到达节点 N-1. 现在给出该有向图,你能告诉windy总共有多少种不同 ...

  7. 【矩阵快速幂】bzoj1297 [SCOI2009]迷路

    1297: [SCOI2009]迷路 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1407  Solved: 1007[Submit][Status ...

  8. [SCOI2009]迷路(矩阵快速幂) 题解

    Description windy在有向图中迷路了. 该有向图有 N 个节点,windy从节点 0 出发,他必须恰好在 T 时刻到达节点 N-1. 现在给出该有向图,你能告诉windy总共有多少种不同 ...

  9. bzoj1297 [SCOI2009]迷路——拆点+矩阵快速幂

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1297 一看感觉是矩阵快速幂之类的,但边权不好处理啊: 普通的矩阵快速幂只能处理边权为1的,所 ...

随机推荐

  1. Redis杂谈

    这是2015年初应邀在南华智闻作技术交流时所作的Redis方面的一个presentation. 因为原件是Keynote格式,已经转成PDF,点击下面链接打开或者下载PDF: Redis 杂谈

  2. casting in C++

    这是2013年写的一篇旧文,放在gegahost.net上面 http://raison.gegahost.net/?p=39 February 20, 2013 casting in C++ Fil ...

  3. pspad的一个怪现象:在一些空行的位置出现个别不该出现的字符

    在用pspad编辑一个外来文件时,发现有许多空行的结尾会出现一些单个字符,字符内容与翻页前那一页相应位置的字符相同. 好奇怪.上网找不到原因.pspad太好用了,不想因此放弃. 仔细观察,这些空行往往 ...

  4. C# string日期格式

    百分数格式应该用“p”这个参数. 格式 原始 数据 结 果 "{0:P}" 0.40 40% 数字 {0:N2} 12.36  数字 {0:N0} 13  货币 {0:c2} $1 ...

  5. log4j 日志分级处理

    log4j 配置文件: log4j.rootLogger=debug,stdout,debug,info,errorlog4j.appender.stdout=org.apache.log4j.Con ...

  6. execve - 执行程序

    总览 (SYNOPSIS) #include <unistd.h> int execve (const char *filename, char *const argv [], char ...

  7. Java数据结构和算法(三)--三大排序--冒泡、选择、插入排序

    三大排序在我们刚开始学习编程的时候就接触过,也是刚开始工作笔试会遇到的,后续也会学习希尔.快速排序,这里顺便复习一下 冒泡排序: 步骤: 1.从首位开始,比较首位和右边的索引 2.如果当前位置比右边的 ...

  8. Visual Studio中Radio Button组绑定变量方法(DDX_Radio方法)

    需求描述:Visual Studio 创建的界面程序中又许多 Radio Button,希望这些所有的Radio Button统一绑定到一个变量上,这个变量一旦改变,Radio Button的选中状态 ...

  9. shell $() vs ${}

    reference $( )与` `(反引号)都是用来做命令替换(command substitution)用的 run command 3, 2, 1 command1 $(command2 $(c ...

  10. 使用 Pytorch 实现 skip-gram 的 word2vec

    转载请注明 AIQ - 最专业的机器学习大数据社区  http://www.6aiq.com AIQ 机器学习大数据 知乎专栏 点击关注 链接地址: https://github.com/lonePa ...