●BOZJ 4456 [Zjoi2016]旅行者
题链:
http://www.lydsy.com/JudgeOnline/problem.php?id=4456
题解:
分治好题。
大致做法如下:
对于一开始的矩形区域,过较长边的中点把矩形区域分为两个块。
然后依次以划分线上的点为起点跑最短路(Dijkstra),
尝试以该点为中转点去更新起点和终点都在这个矩形区域内的询问。
显然,如果某个询问的起点和终点在划分线的两侧,那么此时一定可以求出该询问的答案,
(因为其最短路一定会经过划分线上的点)。
那么至于起点和终点都在同侧的询问,那么就递归到更小的区域去求解是否有更优的答案,
(因为其最短路可能不经过当前划分线上的点)。
具体实现建议直接看代码。
(注意在每次Dijkstra前的操作,很巧妙,可以优化时间——我也是学习网上博主的。)
另外时间复杂度分析参见:http://blog.csdn.net/neither_nor/article/details/51733997
(但复杂度分析的最后一步化简我没看太懂,QAQ)
(不得不说,分治代码还真是考代码能力啊,调了老半天)
代码:
#include<queue>
#include<cstdio>
#include<cstring>
#include<iostream>
#define MAXN 30000
#define MAXM 100050
#define INF 0x3f3f3f3f
#define idx(i,j) (i-1)*M+j
using namespace std;
typedef pair<int,int>pii;
struct Query{
int x1,y1,x2,y2,s,t,id;
}q[MAXM];
struct Edge{
int to[MAXM],val[MAXM],nxt[MAXM],head[MAXN],ent;
void Init(){ent=2;}
void Adde(int u,int v,int w){
to[ent]=v; val[ent]=w; nxt[ent]=head[u]; head[u]=ent++;
to[ent]=u; val[ent]=w; nxt[ent]=head[v]; head[v]=ent++;
}
int Next(int i,bool type){
return type?head[i]:nxt[i];
}
}E;
int N,M,Q;
int dis[MAXN],ans[MAXM],dx[MAXN],dy[MAXN];
void Dijkstra(int S,int X1,int Y1,int X2,int Y2,int w){
static bool vis[MAXN]; static int u,v;
static priority_queue<pii,vector<pii>,greater<pii> >H;
for(int i=X1;i<=X2;i++)
for(int j=Y1;j<=Y2;j++){
u=idx(i,j);
w==INF?dis[u]=INF:dis[u]+=w;
vis[u]=0;
}
dis[S]=0; H.push(make_pair(0,S));
while(!H.empty()){
u=H.top().second; H.pop();
if(vis[u]) continue; vis[u]=1;
for(int i=E.Next(u,1);i;i=E.Next(i,0)){
v=E.to[i];
if(dx[v]<X1||dx[v]>X2||dy[v]<Y1||dy[v]>Y2) continue;
if(dis[v]<=dis[u]+E.val[i]) continue;
dis[v]=dis[u]+E.val[i]; H.push(make_pair(dis[v],v));
}
}
}
void Partition(int X1,int Y1,int X2,int Y2,int ql,int qr){
static Query tmp[MAXM];
if(ql>qr||X1>X2||Y1>Y2) return;
if(X2-X1+1<=Y2-Y1+1){
int mid=(Y1+Y2)/2,L=ql,R=qr,S;
dis[idx(X1,mid)]=INF;
for(int i=X1;i<=X2;i++){
S=idx(i,mid);
Dijkstra(S,X1,Y1,X2,Y2,dis[S]);
for(int k=ql;k<=qr;k++)
ans[q[k].id]=min(ans[q[k].id],dis[q[k].s]+dis[q[k].t]);
}
for(int k=ql;k<=qr;k++){
if(q[k].y1<mid&&q[k].y2<mid) tmp[L++]=q[k];
if(q[k].y1>mid&&q[k].y2>mid) tmp[R--]=q[k];
}
for(int i=ql;i<L;i++) q[i]=tmp[i];
for(int i=qr;i>R;i--) q[i]=tmp[i];
Partition(X1,Y1,X2,mid-1,ql,L-1);
Partition(X1,mid+1,X2,Y2,R+1,qr);
}
else{
int mid=(X1+X2)/2,L=ql,R=qr,S;
dis[idx(mid,Y1)]=INF;
for(int j=Y1;j<=Y2;j++){
S=idx(mid,j);
Dijkstra(S,X1,Y1,X2,Y2,dis[S]);
for(int k=ql;k<=qr;k++)
ans[q[k].id]=min(ans[q[k].id],dis[q[k].s]+dis[q[k].t]);
}
for(int k=ql;k<=qr;k++){
if(q[k].x1<mid&&q[k].x2<mid) tmp[L++]=q[k];
if(q[k].x1>mid&&q[k].x2>mid) tmp[R--]=q[k];
}
for(int i=ql;i<L;i++) q[i]=tmp[i];
for(int i=qr;i>R;i--) q[i]=tmp[i];
Partition(X1,Y1,mid-1,Y2,ql,L-1);
Partition(mid+1,Y1,X2,Y2,R+1,qr);
}
}
int main()
{
freopen("tourist.in","r",stdin);
freopen("tourist.out","w",stdout);
E.Init();
scanf("%d%d",&N,&M);
for(int i=1;i<=N;i++)
for(int j=1;j<=M;j++)
dx[idx(i,j)]=i,dy[idx(i,j)]=j;
for(int i=1,w;i<=N;i++)
for(int j=1;j<M;j++)
scanf("%d",&w),E.Adde(idx(i,j),idx(i,j+1),w);
for(int i=1,w;i<N;i++)
for(int j=1;j<=M;j++)
scanf("%d",&w),E.Adde(idx(i,j),idx(i+1,j),w);
scanf("%d",&Q);
for(int i=1;i<=Q;i++){
scanf("%d%d%d%d",&q[i].x1,&q[i].y1,&q[i].x2,&q[i].y2);
q[i].s=idx(q[i].x1,q[i].y1);
q[i].t=idx(q[i].x2,q[i].y2);
q[i].id=i; ans[i]=INF;
}
Partition(1,1,N,M,1,Q);
for(int i=1;i<=Q;i++) printf("%d\n",ans[i]);
return 0;
}
●BOZJ 4456 [Zjoi2016]旅行者的更多相关文章
- 4456: [Zjoi2016]旅行者
4456: [Zjoi2016]旅行者 https://www.lydsy.com/JudgeOnline/problem.php?id=4456 分析: 每次对当前矩阵按长边化一条分治线,然后在对分 ...
- bzoj 4456 [Zjoi2016]旅行者
题面 https://www.lydsy.com/JudgeOnline/problem.php?id=4456 题解 分治 设当前work的区间为(x1,y1,x2,y2) 我们将长边分成两半 不妨 ...
- BZOJ.4456.[ZJOI2016]旅行者(分治 Dijkstra)
题目链接 \(Description\) 给定\(n\times m\)的带边权网格图.\(Q\)次询问从点\((x_i,y_i)\)到点\((x_j,y_j)\)的最短路. \(n\times m\ ...
- [BZOJ4456] [Zjoi2016]旅行者 分治+最短路
4456: [Zjoi2016]旅行者 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 777 Solved: 439[Submit][Status] ...
- P3350 [ZJOI2016]旅行者
题目描述 小Y来到了一个新的城市旅行.她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北的道路,这些道路两两相交形成n*m个路口 (i,j)(1<=i<=n,1&l ...
- bzoj4456: [Zjoi2016]旅行者
题目链接 bzoj4456: [Zjoi2016]旅行者 题解 网格图,对于图分治,每次从中间切垂直于长的那一边, 对于切边上的点做最短路,合并在图两边的答案. 有点卡常 代码 #include< ...
- 【BZOJ4456】[Zjoi2016]旅行者 分治+最短路
[BZOJ4456][Zjoi2016]旅行者 Description 小Y来到了一个新的城市旅行.她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北的道路,这些道路两两相交形 ...
- luogu3350 [ZJOI2016]旅行者
链接 P3350 [ZJOI2016]旅行者 题目大意:给出网格图,求两点之间最短路,多组询问. \(n*m\leq10^5\ \ q\leq 10^5\) 考虑\(CDQ\)分治. 首先把询问离线, ...
- Luogu 3350 [ZJOI2016]旅行者
BZOJ 4456 听若干个大佬讲过$n$遍终于写掉了. 我把时限基本上跑满了2333…… 分治 + 最短路. 首先我们去分治这个矩形格子,找到一条长边把它对半切,对切开的边上的每一个点跑一遍最短路然 ...
随机推荐
- 解决python中flask_sqlalchemy包安装失败的问题
在进行flask_sqlalchemy包的下载安装时出现以下问题: 由图片可看出是编码转换出了问题,找到pip\compat_init_.py文件,打开它并查看第73行,将代码做如下更改并保存: 问题 ...
- Beta冲刺 第二天
Beta冲刺 第二天 1. 昨天的困难 由于前面的冲刺留下的问题很多,而且混乱的代码给我们接下来的完善工作带来了巨大的困难. 2. 今天解决的进度 潘伟靖: 1.对代码进行了review 2.为系统增 ...
- Windows下编译SDL
Windows下编译SDL的理由我就不多说了,无论用VS来编译或调试SDL库都是很方便的.而且SDL源代码中也包含了VC工程,你所要做的只是解压VC工程,进行适当的配置,然后编译.调试. 编译SDL大 ...
- AWS中的Internet 网关
nternet 网关是一种横向扩展.支持冗余且高度可用的 VPC 组件,可实现 VPC 中的实例与 Internet 之间的通信.因此它不会对网络流量造成可用性风险或带宽限制. Internet 网关 ...
- 记一次jar包冲突
题记:永远不要在同一个项目中,引用不同版本的两个jar包,否则,这可能就是一个大坑. 在做网校项目的时候,帮助中心要使用lucene,所以就引入了lucene-5.5.1的包,删掉了原先存在于项目中的 ...
- 前端面试题:JS中的let和var的区别
最近很多前端的朋友去面试被问到let和var的区别,其实阮一峰老师的ES6中已经很详细介绍了let的用法和var的区别.我简单总结一下,以便各位以后面试中使用. ES6 新增了let命令,用来声明局部 ...
- kafka安装使用和遇到的坑
下载安装 参考:https://segmentfault.com/a/1190000012730949 https://kafka.apache.org/quickstart 关闭服务 关闭zoo ...
- 解决IE8下CSS3选择器 :nth-child() 不兼容的问题
1.定义和用法 :nth-child(n) 选择器匹配属于其父元素的第 N 个子元素,不论元素的类型. n 可以是数字.关键词或公式. <ul> <li>1</li> ...
- 以太坊挖矿源码:clique算法
上文我们总结了以太坊最主要的共识算法:ethash算法,本文将重点分析以太坊的另一个共识算法:clique. 关键字:clique,共识算法,puppeth,以太坊地址原理,区块校验,认证结点,POA ...
- spring9——AOP之AspectJ对AOP的实现
从上述的实验中可以看出BeanNameAutoProxyCreator对于AOP的实现已经和完美了,但是还有两点不足之处: 1,对于切面的实现比较麻烦,既不同类型的通知切面要实现不同的接口,而且一个切 ...