luogu3350 [ZJOI2016]旅行者
链接 P3350 [ZJOI2016]旅行者
- 题目大意:给出网格图,求两点之间最短路,多组询问。
- \(n*m\leq10^5\ \ q\leq 10^5\)
- 考虑\(CDQ\)分治。
- 首先把询问离线,对于一个矩阵中的最短路,如果\(u,v\)不在统一侧,那么一定会经过平分线。
- 所以对矩阵分治理,只考虑最短路经过矩阵中线的情况,枚举中线上的任意一点做最短路,用\(Dis_u+Dis_v\)更新这次询问的答案即可。
- 然后把询问划分到左右两边的矩阵中去,分别递归处理即可。
- 类似于整体二分的思想?
- 代码
// luogu-judger-enable-o2
// luogu-judger-enable-o2
// luogu-judger-enable-o2
#include<bits/stdc++.h>
#define R register int
#define ll long long
using namespace std;
const int N=150001;
const int M=200001;
const int inf=2e9;
int n,m,tot,q,u,cnt,nt[M],w[M],to[M],hd[N],ans[N];
int X[M],Y[M],vis[M],Dis[M];
struct Qs{int id,s,t;}Q[N],tmp[N];
vector<int>idx[N];
void link(R f,R t,R d){nt[++cnt]=hd[f],to[cnt]=t,w[cnt]=d,hd[f]=cnt;}
int gi(){
R x=0,k=1;char c=getchar();
while((c<'0'||c>'9')&&c!='-')c=getchar();
if(c=='-')k=-1,c=getchar();
while(c>='0'&&c<='9')x=(x<<3)+(x<<1)+c-'0',c=getchar();
return x*k;
}
struct nd{int i,v;};
int operator < (nd x,nd y){return x.v>y.v;}
priority_queue<nd>G;
void dij(R s,R lx,R ly,R rx,R ry){
while(!G.empty())G.pop();
for(R i=lx;i<=rx;++i)
for(R j=ly;j<=ry;++j)
Dis[idx[i][j]]=inf,vis[idx[i][j]]=0;
G.push((nd){s,0});
while(!G.empty()){
R i=G.top().i,D=G.top().v;G.pop();
if(vis[i])continue;vis[i]=1,Dis[i]=D;
for(R k=hd[i];k;k=nt[k])
if(!vis[to[k]]&&lx<=X[to[k]]&&rx>=X[to[k]]&&ly<=Y[to[k]]&&ry>=Y[to[k]])
if(Dis[to[k]]>Dis[i]+w[k]){
Dis[to[k]]=Dis[i]+w[k];
G.push((nd){to[k],Dis[to[k]]});
}
}
}
void Div(R lx,R rx,R ly,R ry,R Le,R Ri){
if(Le>Ri)return ;
if(rx-lx>ry-rx){
R mid=(lx+rx)>>1;
for(R i=ly;i<=ry;++i){
dij(idx[mid][i],lx,ly,rx,ry);
for(R j=Le;j<=Ri;++j)
ans[Q[j].id]=min(ans[Q[j].id],Dis[Q[j].s]+Dis[Q[j].t]);
}
R tp1=Le-1,tp2=Ri+1;
for(R i=Le;i<=Ri;++i){
R u=Q[i].s,v=Q[i].t;
if(X[u]<mid&&X[v]<mid)tmp[++tp1]=Q[i];
if(X[u]>mid&&X[v]>mid)tmp[--tp2]=Q[i];
}
for(R i=Le;i<=Ri;++i)Q[i]=tmp[i];
Div(lx,mid-1,ly,ry,Le,tp1);
Div(mid+1,rx,ly,ry,tp2,Ri);
}
else {
R mid=(ly+ry)>>1;
for(R i=lx;i<=rx;++i){
dij(idx[i][mid],lx,ly,rx,ry);
for(R j=Le;j<=Ri;++j)
ans[Q[j].id]=min(ans[Q[j].id],Dis[Q[j].s]+Dis[Q[j].t]);
}
R tp1=Le-1,tp2=Ri+1;
for(R i=Le;i<=Ri;++i){
R u=Q[i].s,v=Q[i].t;
if(Y[u]<mid&&Y[v]<mid)tmp[++tp1]=Q[i];
if(Y[u]>mid&&Y[v]>mid)tmp[--tp2]=Q[i];
}
for(R i=Le;i<=Ri;++i)Q[i]=tmp[i];
Div(lx,rx,ly,mid-1,Le,tp1);
Div(lx,rx,mid+1,ry,tp2,Ri);
}
}
int main(){
n=gi(),m=gi();
for(R i=1;i<=n;++i){
idx[i].push_back(0);
for(R j=1;j<=m;++j){
idx[i].push_back(++tot);
X[tot]=i,Y[tot]=j;
}
}
for(R i=1;i<=n;++i)
for(R j=1;j<m;++j){
u=gi();
link(idx[i][j],idx[i][j+1],u);
link(idx[i][j+1],idx[i][j],u);
}
for(R i=1;i<n;++i)
for(R j=1;j<=m;++j){
u=gi();
link(idx[i][j],idx[i+1][j],u);
link(idx[i+1][j],idx[i][j],u);
}
q=gi();
for(R i=1;i<=q;++i){
R u=gi(),v=gi(),x=gi(),y=gi();
Q[i]=(Qs){i,idx[u][v],idx[x][y]};
}
memset(ans,63,sizeof(ans));
Div(1,n,1,m,1,q);
for(R i=1;i<=q;++i)printf("%d\n",ans[i]);
return 0;
}
luogu3350 [ZJOI2016]旅行者的更多相关文章
- Luogu3350 ZJOI2016 旅行者 最短路、分治
传送门 题意:给出一个$N \times M$的网格图,边有边权,$Q$组询问,每组询问$(x_1,y_1)$到$(x_2,y_2)$的最短路.$N \times M \leq 2 \times 10 ...
- bzoj4456: [Zjoi2016]旅行者
题目链接 bzoj4456: [Zjoi2016]旅行者 题解 网格图,对于图分治,每次从中间切垂直于长的那一边, 对于切边上的点做最短路,合并在图两边的答案. 有点卡常 代码 #include< ...
- 4456: [Zjoi2016]旅行者
4456: [Zjoi2016]旅行者 https://www.lydsy.com/JudgeOnline/problem.php?id=4456 分析: 每次对当前矩阵按长边化一条分治线,然后在对分 ...
- [BZOJ4456] [Zjoi2016]旅行者 分治+最短路
4456: [Zjoi2016]旅行者 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 777 Solved: 439[Submit][Status] ...
- 【BZOJ4456】[Zjoi2016]旅行者 分治+最短路
[BZOJ4456][Zjoi2016]旅行者 Description 小Y来到了一个新的城市旅行.她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北的道路,这些道路两两相交形 ...
- P3350 [ZJOI2016]旅行者
题目描述 小Y来到了一个新的城市旅行.她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北的道路,这些道路两两相交形成n*m个路口 (i,j)(1<=i<=n,1&l ...
- ●BOZJ 4456 [Zjoi2016]旅行者
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=4456 题解: 分治好题.大致做法如下:对于一开始的矩形区域,过较长边的中点把矩形区域分为两个 ...
- [ZJOI2016]旅行者
题目描述 小Y来到了一个新的城市旅行.她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北的道路,这些道路两两相交形成n*m个路口 (i,j)(1<=i<=n,1&l ...
- BZOJ4456/UOJ#184[Zjoi2016]旅行者 分治 最短路
原文链接http://www.cnblogs.com/zhouzhendong/p/8682133.html 题目传送门 - BZOJ4456 题目传送门 - UOJ#184 题意 $n\times ...
随机推荐
- Spark Streaming的优化之路—从Receiver到Direct模式
作者:个推数据研发工程师 学长 1 业务背景 随着大数据的快速发展,业务场景越来越复杂,离线式的批处理框架MapReduce已经不能满足业务,大量的场景需要实时的数据处理结果来进行分析.决 ...
- 源码编译安装Apache/2.4.37-------踩了无数坑,重装了十几次服务器才会的,不容易啊!
1.先进入/usr/local/中创建三个文件夹 apr apr-util apache cd /usr/local目录 mkdir apr mkdir apr-util mkdir apache 2 ...
- qbzt day2 上午
内容提要 贪心 分治 分块 搜索 接着昨天的讲 过河问题 考虑AB是最快的人,CD是最慢的人,要把CD两个人送过河,只有两种方案,牵扯到四个人,并且n个规模的原问题化成了n-2个规模的子问题 那么最后 ...
- VMware 虚拟化编程(12) — VixDiskLib Sample 程序使用
目录 目录 前文列表 vixDiskLibSample 安装 Sample 程序 Sample 程序使用方法 前文列表 VMware 虚拟化编程(1) - VMDK/VDDK/VixDiskLib/V ...
- 【ABAP系列】SAP MB5B中FI凭证摘要是激活的/结果可能不正确 的错误
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[ABAP系列]SAP MB5B中FI凭证摘要是 ...
- Good teachers,they inspire you, they entertain you,and you end up learning a ton even when you don't know it.
pardon. v. 原谅.抱歉.再说一次 honourable.adj.值得钦佩的 specification.n.规格.标准 amongst.prep.在...中 gallon.n.加仑 comp ...
- == 和 equals的区别
== 和 equals的区别 基本类型:== 比较的是两个变量的面值大小 对象对象: 比较的是内存地址 特例: String a = "abc" String b = &qu ...
- <编译原理 - 函数绘图语言解释器(3)解释器 - python>
<编译原理 - 函数绘图语言解释器(3)解释器 - python> <编译原理 - 函数绘图语言解释器(2)词法分析器 - python> <编译原理 - 函数绘图语言解 ...
- Mac入门--通过Homebrew安装PHP(新)
1 首先安装homebrew,安装过的话更新 安装:homebrew官网地址:https://brew.sh/index_zh-cn.html.或者直接复制下面代码: /usr/bin/ruby -e ...
- Linq查询语法(1)
转:http://www.cnblogs.com/ahao214/archive/2013/01/22/2871044.html LINQ的基本格式如下所示:var <变量> = from ...