[难题题解] [BZOJ1875] [SDOI2009] HH去散步
题目H有个一成不变的习惯,喜欢饭后百步走。所谓百步走,就是散步,就是在一定的时间 内,走过一定的距离。 但是同时HH又是个喜欢变化的人,所以他不会立刻沿着刚刚走来的路走回。 又因为HH是个喜欢变化的人,所以他每天走过的路径都不完全一样,他想知道他究竟有多 少种散步的方法。 现在给你学校的地图(假设每条路的长度都是一样的都是1),问长度为t,从给定地 点A走到给定地点B共有多少条符合条件的路径.
对于100%的数据,$N ≤ 20,M ≤ 60,t ≤ 2^{30},0 ≤ A,B $
题解
既然n<=20,考虑开个邻接矩阵$table$存图
先考虑暴力,设$dp[i][j]$表示还需走i段路,当前走到了j这个点
易得$dp[i][j]=sum(dp[i+1][k]*table[j][k])+1$
然后无脑dfs即可
#include <iostream>
#include <cstdio>
#include <vector>
using namespace std;
int table[21][21],dp[21][11],n,a,b;
void dfs(int id,int from,int t)
{
//if(dp[id][from]) return;
if(!t)
{
dp[id][t]+=(id==b);
return;
}
for(int i=0;i<n;i++)
{
if(!table[id][i]||from==i) continue;
dfs(i,id,t-1);
dp[id][t]+=table[id][i]*dp[i][t-1];
dp[id][t]%=45989;
}
}
int main()
{
int m,t;
cin>>n>>m>>t>>a>>b;
for(int i=1;i<=m;i++)
{
int a,b;
scanf("%d%d",&a,&b);
table[a][b]=++table[b][a];
}
dfs(a,0,t);
cout<<dp[b][0]; }
发现i这一维可以滚动掉,然后我们发现这个方程的转移是固定不变
即,对于同一个j,都是同一批k来更新它。
那么,我们可以先处理一个另一个邻接矩阵$table2[k][i][j]$表示i在走了k步后到达j的方案书.
回忆floyd的过程,我们可以用类似的方法不断用$table2[k-1][i][j]$计算出新的$table2[k][i][j]$
然后发现其实这就是矩阵乘法,于是这个过程可以用矩阵快速幂来加速。
这其实是一个很常见的图论dp trick,当然,这个是要在点数小到可以用邻接矩阵时才能用的
void mulit(int arr1[][130],int arr2[][130],int size)
{
memset(temp,0,sizeof(temp));
for(int i=1;i<=size;i++)
{
for(int j=1;j<=cnt;j++)
{
for(int k=1;k<=cnt;k++)
{
temp[i][j]+=arr1[i][k]*arr2[k][j]%mod;
temp[i][j]%=mod;
}
}
}
memcpy(arr1,temp,sizeof(temp));
}
void qpow(int t)
{
for(int i=1;i<=cnt;i++) res[i][i]=1;
while(t)
{
if(t&1) mulit(res,table2,cnt);
mulit(table2,table2,cnt);
t>>=1;
}
}
然而题目要求我们不能立刻走回头路,处理起来比较麻烦
所以我们把边当成点,把相连的边“连接”
另外,为了区分出边的方向,我们把每条边拆成两条(分别两个方向)
然后按照上述的方法处理出$table2[t][i][j]$
最后枚举一遍从起点发出的边,统计这些边到达终点的方案数,输出即可
代码
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int table2[130][130],n,a,b,cnt,from[130],to[130],head[130],nxt[130];
int res[130][130];
#define connect(a,b) table2[a][b]=1
#define mod 45989
int temp[130][130];
void mulit(int arr1[][130],int arr2[][130],int size)
{
memset(temp,0,sizeof(temp));
for(int i=1;i<=size;i++)
{
for(int j=1;j<=cnt;j++)
{
for(int k=1;k<=cnt;k++)
{
temp[i][j]+=arr1[i][k]*arr2[k][j]%mod;
temp[i][j]%=mod;
}
}
}
memcpy(arr1,temp,sizeof(temp));
}
void qpow(int t)
{
for(int i=1;i<=cnt;i++) res[i][i]=1;
while(t)
{
if(t&1) mulit(res,table2,cnt);
mulit(table2,table2,cnt);
t>>=1;
}
}
void link(int x,int y)
{
from[++cnt]=x,to[cnt]=y;
nxt[cnt]=head[x];
head[x]=cnt;
}
int main()
{
int m,t;
cin>>n>>m>>t>>a>>b;
a++,b++;
for(int i=1;i<=m;i++)
{
int x,y;
scanf("%d%d",&x,&y);
x++,y++;
link(x,y);
link(y,x);
}
for(int i=1;i<=cnt;i++)
{
for(int j=1;j<=cnt;j++)
{
if(((i&1)&&j==i+1)||((i&1)==0&&j==i-1)) continue;
if(to[i]==from[j]) connect(i,j);
//if(from[i]==to[j]) connect(j,i); }
}
/*for(int i=1;i<=cnt;i++)
{
for(int j=1;j<=cnt;j++) cout<<table2[i][j]<<" ";
cout<<endl;
}*/
qpow(t-1);
int ans=0;
for(int l=head[a];l;l=nxt[l])
for(int i=1;i<=cnt;i++)
if(to[i]==b) ans+=res[l][i],ans%=mod;
cout<<ans;
}
[难题题解] [BZOJ1875] [SDOI2009] HH去散步的更多相关文章
- bzoj1875: [SDOI2009]HH去散步
终于A了...早上按自己以前的写法一直WA.下午换了一种写法就A了qwq #include<cstdio> #include<cstring> #include<iost ...
- 【题解】 bzoj1875: [SDOI2009]HH去散步 (动态规划+矩阵乘法)
bzoj1875,懒得复制,戳我戳我 Solution: 看到这道题,看的出是个dp,每个点\(t\)时刻到达的方案数等于\(t-1\)到连过来的点方案数之和 但又因为题目有要求不能走一样的边回去不是 ...
- BZOJ1875 [SDOI2009]HH去散步 矩阵
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1875 题意概括 在一个无向图(有重边无自环)中走,不能在经过连续经过某一条边2次. 现在走t步,问 ...
- BZOJ1875 [SDOI2009]HH去散步 【dp + 矩阵优化】
题目 HH有个一成不变的习惯,喜欢饭后百步走.所谓百步走,就是散步,就是在一定的时间 内,走过一定的距离. 但 是同时HH又是个喜欢变化的人,所以他不会立刻沿着刚刚走来的路走回. 又因为HH是个喜欢变 ...
- bzoj1875 [SDOI2009]HH去散步 矩阵快速幂
题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=1875 题解 如果没有这个"不能立刻沿着刚刚走来的路走回",那么这个题就是一 ...
- [bzoj1875][SDOI2009] HH去散步 [dp+矩阵快速幂]
题面 传送门 正文 其实就是让你求有多少条长度为t的路径,但是有一个特殊条件:不能走过一条边以后又立刻反着走一次(如果两次经过同意条边中间隔了别的边是可以的) 如果没有这个特殊条件,我们很容易想到dp ...
- BZOJ1875: [SDOI2009]HH去散步 图上边矩乘
这道题十分的坑…… 我作为一只连矩乘都不太会的渣渣看到这道题就只能神搜了….. 首先说一下普通的矩乘求方案,就是高出邻接矩阵然后一顿快速幂….. 矩乘一般就是一些秘制递推….. 再说一下这道题,我们可 ...
- bzoj1875 [SDOI2009]HH去散步——矩阵快速幂
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1875 有个限制是不能走回头路,比较麻烦: 所以把矩阵中的元素设成边的经过次数,单向边之间就好 ...
- bzoj 1875: [SDOI2009]HH去散步 -- 矩阵乘法
1875: [SDOI2009]HH去散步 Time Limit: 20 Sec Memory Limit: 64 MB Description HH有个一成不变的习惯,喜欢饭后百步走.所谓百步走, ...
随机推荐
- SpringBoot--使用Spring Cache整合redis
一.简介 Spring Cache是Spring对缓存的封装,适用于 EHCache.Redis.Guava等缓存技术. 二.作用 主要是可以使用注解的方式来处理缓存,例如,我们使用redis缓存时, ...
- 致Spring Boot初学者
1.引言 Spring Boot是近两年来火的一塌糊涂,来这里的每一位同学,之前应该大致上学习了web项目开发方面的知识,正在努力成长过程中.因为最近有不少人来向我“请教”,他们大都是一些刚入门的新手 ...
- python中的多任务--线程
什么是多任务? 简单地说,就是操作系统可以同时运行多个任务. 实现多任务有多种方式,线程.进程.协程. 多任务的概念:并行和并发 并发:指的是任务数多余cpu核数,通过操作系统的各种任务调度算法, 实 ...
- JavaScript动画实例:递归分形图动态展示
在“JavaScript图形实例:SierPinski三角形” 和“JavaScript图形实例:Levy曲线及其变形”等文章中我们介绍了通过递归生成分形图形的方法.我们可以将绘制的分形图形每隔一定的 ...
- 极致Web性能 —— SPA性能指南
前言 前端框架时代,为开发体验.效率与页面性能带来,非常大的革命.大家纷纷拿起一系列打包工具(webpack/parcel etc.),配合一系列加载器快速搭建起一个 SPA 页面. SPA 应用带来 ...
- 揭秘JAVA JVM内幕
在之前的文章 一步步解析java执行内幕 中,比较详细分析了java代码是如何一步一步在jvm中执行的,然而设计的的jvm核心技术点,并未做深入分析,本篇文章将重点分析jvm,涉及到的内容包括jvm内 ...
- AHP(使用于某项目设备重要度评估测试)
用层次法和蒙特卡洛模型计算权重系数,然后建立判断矩阵进行随机一致性检验,最后求出重要度指数. string calculateStr = "1,2,3,2,1,|1,2,3,2,1,|1,2 ...
- 重学c#系列——对c#粗浅的认识(一)
前言 什么是c#呢? 首先你是如何读c#的呢?c sharp?或者c 井? 官方读法是:see sharp. 有没有发现开发多年,然后感觉名字不对. tip:为个人重新整理,如学习还是看官网,c# 文 ...
- 方正璞华Java面试总结(武汉)
方正璞华Java面试总结(武汉) 现在社会急缺复合型人才,计算机与日语的结合,具备这两种能力的人不愁工作,最后他们大多到的也是日企,甚至到日本去工作.至今为止接触的日企有光庭.方正璞华.先锋·商泰.英 ...
- MYSQL 之 JDBC(十五):数据库连接池
在使用开发基于数据库的web程序时,传统的模式基本是按一下步骤: 在主程序(如servlet.bean)中建立数据库连接 进行sql操作 断开数据库连接 这种模式开发存在各种各样的问题,最重要的是:数 ...