洛谷 P6573 [BalticOI 2017] Toll 题解
算是回归OI后第一道自己写的题(考CSP的时候可没回归)
写篇题解纪念一下
题目大意: \(n\) 个点,\(m\) 条单向边,每条边的两端点 \(x\),\(y\)必定满足 \(\left\lfloor\dfrac{y}{k}\right\rfloor=\left\lfloor\dfrac{x}{k}\right\rfloor+1\), \(q\)次询问,求每一对\(a\),\(b\)的最短距离。
\(n<=5*10^{4},q<=10^4\)\(1<k<5\),保证\(a\)<\(b\)
简单分析一下容易发现这是个分层图,每 \(k\) 个点为一层,只有相邻的层间有边且边只能往编号大的方向去。
然后就是考虑这个东西怎么维护了,考虑前两天CSP2022的T4做法之一,矩阵乘法+倍增优化dp
容易发现我们可以把每个层缩成一个点,之后的图就成了好多条链(注意不一定保证联通)
然后对于这条链中的每两个相邻节点(即相邻两层),我们可以用类似Floyd的方法把原图中的边构造一个矩阵,接着就可以矩阵乘法快速合并了。
然后再倍增处理一下就结束了
记得判连通性。
(我自己的实现把链当成树了所以看起来有些麻烦。。。但是问题不大,能过就行)
点击查看代码
#include <bits/stdc++.h>
#define N 50010
#define M 1000010
#define pii pair<int,int>
#define mkp make_pair
#define pb push_back
#define fi first
#define se second
//#define int long long
//#define MOD
#define INF 1000000000
#define int_edge int to[M],nxt[M],head[N],cnt=0;
using namespace std;
int k,n,m,q,fa[N][16],f[N],dep[N],eid=0,vis[N];
int_edge;void add_edge(int x,int y ){to[++cnt]=y;nxt[cnt]=head[x];head[x]=cnt;}
map<pii,int>mp;
struct Y{int x,y,t;};vector<Y>E[N];
int id(int x){return x/k+1;}
//int_edge;int val[M];void add_edge(int x,int y,int z){to[++cnt]=y;val[cnt]=z;nxt[cnt]=head[x];head[x]=cnt;}
struct Matrix{
int a[5][5];
Matrix(){memset(a,0x3f,sizeof(a));}
Matrix operator * (const Matrix &y){
Matrix sum;
for(int i=0;i<5;i++)
for(int j=0;j<5;j++)
for(int k=0;k<5;k++)
sum.a[i][j]=min(sum.a[i][j],a[i][k]+y.a[k][j]);
return sum;
}
}dn[N][16];
Matrix init(){
Matrix x;for(int i=0;i<k;i++)x.a[i][i]=0;return x;
}
int find(int x){return x==f[x]?x:f[x]=find(f[x]);}
void dfs(int nw,int lst){
vis[nw]=1;
fa[nw][0]=lst;dep[nw]=dep[lst]+1;
if(lst!=0){
for(auto e:E[mp[mkp(lst,nw)]]){
dn[nw][0].a[e.x%k][e.y%k]=e.t;
}
}
for(int i=1;i<=15;i++){
fa[nw][i]=fa[fa[nw][i-1]][i-1];
dn[nw][i]=dn[fa[nw][i-1]][i-1]*dn[nw][i-1];
}
for(int i=head[nw];i;i=nxt[i])
if(to[i]!=lst)dfs(to[i],nw);
}
signed main()
{
scanf("%d %d %d %d",&k,&n,&m,&q);
for(int i=0;i<n;i++)f[i]=i;
for(int i=1,x,y,t;i<=m;i++){
scanf("%d %d %d",&x,&y,&t);
f[find(x)]=find(y);
if(mp.find(mkp(id(x),id(y)))==mp.end()){
mp[mkp(id(x),id(y))]=++eid;
add_edge(id(x),id(y));
}
E[mp[mkp(id(x),id(y))]].pb(Y{x,y,t});
}
for(int i=1;i<=id(n);i++)if(!vis[i])dfs(i,0);
while(q--){
int x,y;scanf("%d %d",&x,&y);
if(find(x)!=find(y)){puts("-1");continue;}
int id1=id(x),id2=id(y);
Matrix ans=init();
for(int i=15;i>=0;i--)
if(dep[fa[id2][i]]>=dep[id1])
ans=dn[id2][i]*ans,id2=fa[id2][i];
if(ans.a[x%k][y%k]==1061109567)puts("-1");
else printf("%d\n",ans.a[x%k][y%k]);
}
return 0;
}
洛谷 P6573 [BalticOI 2017] Toll 题解的更多相关文章
- 洛谷P1783 海滩防御 分析+题解代码
洛谷P1783 海滩防御 分析+题解代码 题目描述: WLP同学最近迷上了一款网络联机对战游戏(终于知道为毛JOHNKRAM每天刷洛谷效率那么低了),但是他却为了这个游戏很苦恼,因为他在海边的造船厂和 ...
- 洛谷P4047 [JSOI2010]部落划分题解
洛谷P4047 [JSOI2010]部落划分题解 题目描述 聪聪研究发现,荒岛野人总是过着群居的生活,但是,并不是整个荒岛上的所有野人都属于同一个部落,野人们总是拉帮结派形成属于自己的部落,不同的部落 ...
- 洛谷P1155 双栈排序题解(图论模型转换+二分图染色+栈)
洛谷P1155 双栈排序题解(图论模型转换+二分图染色+栈) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1311990 原题地址:洛谷P1155 双栈排序 ...
- 洛谷 P4665 [BalticOI 2015]Network
洛谷 P4665 [BalticOI 2015]Network 你有一棵 $ n $ 个节点的树,你可以在树上加一些边,使这棵树变成一张无重边.自环的图,且删掉任意一条边它仍然联通.求最少要加多少条边 ...
- 洛谷 P3951 NOIP 2017 小凯的疑惑
洛谷 P3951 NOIP 2017 小凯的疑惑 题目描述 小凯手中有两种面值的金币,两种面值均为正整数且彼此互素.每种金币小凯都有 无数个.在不找零的情况下,仅凭这两种金币,有些物品他是无法准确支付 ...
- 洛谷10月月赛II题解
[咻咻咻] (https://www.luogu.org/contestnew/show/11616) 令人窒息的洛谷月赛,即将参加NOIp的我竟然只会一道题(也可以说一道也不会),最终145的我只能 ...
- [洛谷P1823]音乐会的等待 题解(单调栈)
[洛谷P1823]音乐会的等待 Description N个人正在排队进入一个音乐会.人们等得很无聊,于是他们开始转来转去,想在队伍里寻找自己的熟人.队列中任意两个人A和B,如果他们是相邻或他们之间没 ...
- BZOJ2527 & 洛谷3527:[Poi2011]Meteors——题解
+++++++++++++++++++++++++++++++++++++++++++ +本文作者:luyouqi233. + +欢迎访问我的博客:http://www.cnblogs.com/luy ...
- 洛谷 p1516 青蛙的约会 题解
dalao们真是太强了,吊打我无名蒟蒻 我连题解都看不懂,在此篇题解中,我尽量用语言描述,不用公式推导(dalao喜欢看公式的话绕道,这篇题解留给像我一样弱的) 进入正题 如果不会扩展欧里几德的话请先 ...
随机推荐
- CCPC比赛与算法学习的个人分享
大赛简介 中国大学生程序设计竞赛(China Collegiate Programming Contest,简称CCPC)是工业和信息化部教育与考试中心主办的 "强国杯"技术技能大 ...
- 【ARK UI】HarmonyOS 从相册选择图片并显示到Image组件上
参考资料 [Harmony OS][ARK UI]ETS 上下文基本操作 [Harmony OS][ARK UI]ets使用startAbility或startAbilityForResult方式 ...
- 【JAVA】学习路径64-补充-编写一个会抛异常的方法
有一些方法,在调用的时候有可能会出错,所以我们使用这些方法的时候会使用try catch. 比如InputStream里面的read()方法等等,那么这些方法是怎么实现抛异常的效果的呢? 能抛异常的方 ...
- No value specified for parameter 5异常
No value specified for parameter 5 翻译:没有为参数5指定值 在sql语句中,有5个" ? "号,但是赋值的时候只赋了前面4个" ?&q ...
- 第一百篇:JS异步
好家伙,打工人要打工,博客会更新的没有以前频繁了 芜湖,一百篇了,这篇写一个比较难的异步(其实并不难理解,主要是为promise铺垫) 老样子,先补点基础: 1.进程 来吧,新华字典 大 ...
- ubuntu 16.04 搭建docker
1.为确保运行的不是旧版Docker sudo apt-get remove docker docker-engine docker.io 2.添加Docker COmmunity Edition 所 ...
- KingbaseES timestamp 和 timestamptz 差异比较
KingbaseES 提供两种存储时间戳的数据类型: 不带时区的 TIMESTAMP 和带时区的 TIMESTAMPTZ. TIMESTAMP 数据类型可以同时存储日期和时间,但它不存储时区.这意味着 ...
- gin如何多次shoubind一个请求参数
gin多次绑定请求参数 package main import ( "fmt" "net/http" "time" "github ...
- [机器学习]-分类问题常用评价指标、混淆矩阵及ROC曲线绘制方法
分类问题 分类问题是人工智能领域中最常见的一类问题之一,掌握合适的评价指标,对模型进行恰当的评价,是至关重要的. 同样地,分割问题是像素级别的分类,除了mAcc.mIoU之外,也可以采用分类问题的一些 ...
- 《吐血整理》高级系列教程-吃透Fiddler抓包教程(23)-Fiddler如何优雅地在正式和测试环境之间来回切换-上篇
1.简介 在开发或者测试的过程中,由于项目环境比较多,往往需要来来回回地反复切换,那么如何优雅地切换呢?宏哥今天介绍几种方法供小伙伴或者童鞋们进行参考. 2.实际工作场景 2.1问题场景 (1)已发布 ...