首页 > 代码库 > HH去散步[SDOI2009]

HH去散步[SDOI2009]

题目描述

HH有个一成不变的习惯,喜欢饭后百步走。所谓百步走,就是散步,就是在一定的时间 内,走过一定的距离。 但是同时HH又是个喜欢变化的人,所以他不会立刻沿着刚刚走来的路走回。 又因为HH是个喜欢变化的人,所以他每天走过的路径都不完全一样,他想知道他究竟有多 少种散步的方法。 现在给你学校的地图(假设每条路的长度都是一样的都是1),问长度为t,从给定地 点A走到给定地点B共有多少条符合条件的路径

输入

第一行:五个整数N,M,t,A,B。
N表示学校里的路口的个数
M表示学校里的 路的条数
t表示HH想要散步的距离
A表示散步的出发点
B则表示散步的终点。
接下来M行
每行一组Ai,Bi,表示从路口Ai到路口Bi有一条路。
数据保证Ai != Bi,但不保证任意两个路口之间至多只有一条路相连接。 
路口编号从0到N -1。 
同一行内所有数据均由一个空格隔开,行首行尾没有多余空格。没有多余空行。 
答案模45989。
N ≤ 20,M ≤ 60,t ≤ 2^30,0 ≤ A,B

输出

一行,表示答案。

样例输入

4 5 3 0 0
0 1
0 2
0 3
2 1
3 2

样例输出

4

题解
    考试的时候想得比较粗略,一开始当然是暴力深搜,也知道这种方案数还取模的题不可能深搜出正解,但还是先打了一通,没费什么力气就出了样例。之后开始想正解,感觉或许应该把环收缩一下,但是环环相扣环环重叠不知道应该怎么处理,最后还是把深搜交上去了。
    正解是矩阵优化dp(但总感觉它其实说不上是dp),比如矩阵G[i,j]的k次方中的g(a,b)可以表示从a点到b点经过k条边的方案数,本题因为对于边有特殊的要求(不立刻反向)所以把边放进矩阵,G[i,j]为1表示从i到j有一条边,这里的边是分方向的,双向建边就会有2*m条。tot矩阵(其实只有一行)表示从起点有哪些边出来。tot乘上G[i,j]的k-1次方(注意顺序,矩阵乘法不满足交换律),把得到的一行结果矩阵中有边到终点的位加和,得到的结果即为答案。
    刚开始觉得矩阵非常抽象,虽然自己在课下学过很多数学书上的理论知识但还是不会用,不过这道题让我明白矩阵也是有明确含义的(加速矩阵除外),本质上还是要理解算法每一步的目的。

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int n,m,t,a,b,temp;
const int md=45989;
int dis[125][125]={0},tot[2][125]={0},ans[125][125]={0};
inline int r()
{
    int jg=0,jk=0;
    jk=getchar()-0;
    if(jk>=0&&jk<=9)  jg+=jk;
    jk=getchar()-0;
    while(jk>=0&&jk<=9)
    {
       jg*=10;
       jg+=jk;
       jk=getchar()-0;
    }
    return jg;
}
int h[25],e;
struct B
{
     int u,v,ne;
}bi[125];
void add(int x,int y)
{
     bi[e].u=x;
     bi[e].v=y;
     bi[e].ne=h[x];
     h[x]=e++;
}
int jg[125][125];
void jc(int cs1[][125],int cs2[][125])
{
     memset(jg,0,sizeof(jg));
     for(int i=0;i<temp;i++)
       for(int j=0;j<temp;j++)
         for(int k=0;k<temp;k++)
           jg[i][j]+=(cs1[i][k]*cs2[k][j])%md;
     for(int i=0;i<temp;i++)
       for(int j=0;j<temp;j++)
         cs1[i][j]=jg[i][j]%md;
}
void init()
{
     n=r();
     m=r();
     t=r();
     a=r();
     b=r();
     temp=2*m;
     memset(h,-1,sizeof(h));
     int a1,a2;
     for(int i=0;i<m;i++)
     {
        a1=r();
        a2=r();
        add(a1,a2);
        add(a2,a1);
     }
     for(int i=0;i<n;i++)
      for(int j=h[i];j!=-1;j=bi[j].ne)
        for(int k=h[bi[j].v];k!=-1;k=bi[k].ne)
        {
          if(((k&1)&&k==j+1)||((j&1)&&j==k+1))
            continue;
          dis[j][k]=1;
        }
     for(int i=h[a];i!=-1;i=bi[i].ne)
          tot[0][i]=1;
}
int main()
{
    init();
    for(int i=0;i<temp;i++)
      ans[i][i]=1;
    t--;
    while(t)
    {
       if(t&1) jc(ans,dis);
       t>>=1;
       jc(dis,dis);
    }
    jc(tot,ans);
    int res=0;
    for(int i=h[b];i!=-1;i=bi[i].ne)
    {
      if(i&1)
        res+=tot[0][i-1];
      else
        res+=tot[0][i+1];
    }
    printf("%d",res%md);
    return 0;
}

 

 

HH去散步[SDOI2009]