感觉比较套路,每次在长边中轴线处切一刀,求出切割线上的点对矩形内所有点的单源最短路径,以此更新每个询问,递归处理更小的矩形。因为若起点终点跨过中轴线是肯定要经过的,而不跨过中轴线的则可以选择是否经过中轴线,若不经过一定就在矩形的某一半了。复杂度O((nm)1.5log(nm)),不太会证。

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
int read()
{
int x=,f=;char c=getchar();
while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
return x*f;
}
#define N 20010
#define M 200
#define Q 100010
int n,m,u,op,a[N*M][],dis[M][N],ans[Q];
struct data{int sx,sy,tx,ty,i;
}q[Q],tmp[Q];
int wx[]={-,,,},wy[]={,,,-};
int trans(int x,int y){return (x-)*m+y;}
namespace shortestpath
{
int p[N],t;bool flag[N];
struct data{int to,nxt,len;}edge[N<<];
struct data2
{
int x,d;
bool operator <(const data2&a) const
{
return d>a.d;
}
};
priority_queue<data2> q;
void addedge(int x,int y,int z){t++;edge[t].to=y,edge[t].nxt=p[x],edge[t].len=z,p[x]=t;}
void dijkstra(int x,int y,int k)
{
while (!q.empty()) q.pop();
memset(dis[k],,sizeof(dis[k]));dis[k][trans(x,y)]=;q.push((data2){trans(x,y),});
memset(flag,,sizeof(flag));
while ()
{
while (!q.empty()&&flag[q.top().x]) q.pop();
if (q.empty()) break;
data2 v=q.top();q.pop();
flag[v.x]=;
for (int j=p[v.x];j;j=edge[j].nxt)
if (v.d+edge[j].len<dis[k][edge[j].to])
{
dis[k][edge[j].to]=v.d+edge[j].len;
q.push((data2){edge[j].to,dis[k][edge[j].to]});
}
}
}
void make(int u,int d,int l,int r)
{
t=;
for (int i=u;i<=d;i++)
for (int j=l;j<=r;j++)
p[trans(i,j)]=;
for (int i=u;i<=d;i++)
for (int j=l;j<=r;j++)
for (int k=;k<;k++)
if (i+wx[k]>=u&&i+wx[k]<=d&&j+wy[k]>=l&&j+wy[k]<=r)
addedge(trans(i,j),trans(i+wx[k],j+wy[k]),a[trans(i,j)][k]);
}
}
void solve(int u,int d,int l,int r,int x,int y)
{
if (u>d||l>r||x>y) return;
shortestpath::make(u,d,l,r);
if (d-u<=r-l)
{
int mid=l+r>>,s=x-,t=y+;
for (int i=u;i<=d;i++) shortestpath::dijkstra(i,mid,i-u+);
for (int i=x;i<=y;i++)
{
for (int j=;j<=d-u+;j++)
ans[q[i].i]=min(ans[q[i].i],dis[j][trans(q[i].sx,q[i].sy)]+dis[j][trans(q[i].tx,q[i].ty)]);
if (max(q[i].sy,q[i].ty)<mid) tmp[++s]=q[i];
else if (min(q[i].sy,q[i].ty)>mid) tmp[--t]=q[i];
}
for (int i=x;i<=s;i++) q[i]=tmp[i];
for (int i=t;i<=y;i++) q[i]=tmp[i];
solve(u,d,l,mid-,x,s),
solve(u,d,mid+,r,t,y);
}
else
{
int mid=u+d>>,s=x-,t=y+;
for (int i=l;i<=r;i++) shortestpath::dijkstra(mid,i,i-l+);
for (int i=x;i<=y;i++)
{
for (int j=;j<=r-l+;j++)
ans[q[i].i]=min(ans[q[i].i],dis[j][trans(q[i].sx,q[i].sy)]+dis[j][trans(q[i].tx,q[i].ty)]);
if (max(q[i].sx,q[i].tx)<mid) tmp[++s]=q[i];
else if (min(q[i].sx,q[i].tx)>mid) tmp[--t]=q[i];
}
for (int i=x;i<=s;i++) q[i]=tmp[i];
for (int i=t;i<=y;i++) q[i]=tmp[i];
solve(u,mid-,l,r,x,s),
solve(mid+,d,l,r,t,y);
}
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("bzoj4456.in","r",stdin);
freopen("bzoj4456.out","w",stdout);
const char LL[]="%I64d\n";
#else
const char LL[]="%lld\n";
#endif
n=read(),m=read();
for (int i=;i<=n;i++)
for (int j=;j<m;j++)
a[trans(i,j)][]=a[trans(i,j+)][]=read();
for (int i=;i<n;i++)
for (int j=;j<=m;j++)
a[trans(i,j)][]=a[trans(i+,j)][]=read();
u=read();
for (int i=;i<=u;i++)
q[i].sx=read(),q[i].sy=read(),q[i].tx=read(),q[i].ty=read(),q[i].i=i;
memset(ans,,sizeof(ans));
solve(,n,,m,,u);
for (int i=;i<=u;i++) printf("%d\n",ans[i]);
return ;
}

BZOJ4456 ZJOI2016旅行者(分治+最短路)的更多相关文章

  1. [BZOJ4456] [Zjoi2016]旅行者 分治+最短路

    4456: [Zjoi2016]旅行者 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 777  Solved: 439[Submit][Status] ...

  2. 【BZOJ4456】[Zjoi2016]旅行者 分治+最短路

    [BZOJ4456][Zjoi2016]旅行者 Description 小Y来到了一个新的城市旅行.她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北的道路,这些道路两两相交形 ...

  3. 【BZOJ-4456】旅行者 分治 + 最短路

    4456: [Zjoi2016]旅行者 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 254  Solved: 162[Submit][Status] ...

  4. BZOJ4456/UOJ#184[Zjoi2016]旅行者 分治 最短路

    原文链接http://www.cnblogs.com/zhouzhendong/p/8682133.html 题目传送门 - BZOJ4456 题目传送门 - UOJ#184 题意 $n\times ...

  5. 【BZOJ4456】旅行者(最短路,分治)

    [BZOJ4456]旅行者(最短路,分治) 题面 BZOJ Description 小Y来到了一个新的城市旅行.她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北 的道路,这些 ...

  6. bzoj4456: [Zjoi2016]旅行者

    题目链接 bzoj4456: [Zjoi2016]旅行者 题解 网格图,对于图分治,每次从中间切垂直于长的那一边, 对于切边上的点做最短路,合并在图两边的答案. 有点卡常 代码 #include< ...

  7. [BZOJ4456][ZJOI2016]旅行者:分治+最短路

    分析 类似于点分治的思想,只统计经过分割线的最短路,然后把地图一分为二. 代码 #include <bits/stdc++.h> #define rin(i,a,b) for(regist ...

  8. BZOJ.4456.[ZJOI2016]旅行者(分治 Dijkstra)

    题目链接 \(Description\) 给定\(n\times m\)的带边权网格图.\(Q\)次询问从点\((x_i,y_i)\)到点\((x_j,y_j)\)的最短路. \(n\times m\ ...

  9. 4456: [Zjoi2016]旅行者

    4456: [Zjoi2016]旅行者 https://www.lydsy.com/JudgeOnline/problem.php?id=4456 分析: 每次对当前矩阵按长边化一条分治线,然后在对分 ...

随机推荐

  1. 编译安装开源免费中文分词scws

    一.SCWS了解一下: SCWS 是 Simple Chinese Word Segmentation 的首字母缩写(即:简易中文分词系统). 这是一套基于词频词典的机械式中文分词引擎,它能将一整段的 ...

  2. python的列表数据类型及常用操作

    列表是最常用的Python数据类型,它可以作为一个方括号内的逗号分隔值出现. 列表中的每个元素都分配一个数字 - 它的位置,或索引,第一个索引是0,第二个索引是1,依此类推. 列表可以进行的操作包括索 ...

  3. jQuery的封装

    封装,最简单的效果就是一个效果你可以重复的去调用   本来前端之路时间不是很长,但是对封装还是有一点点自己个人的理解,曾经踩过的坑也不在少数,最后总结出我个人风格的封装,听一位大神指点过,每个人都有属 ...

  4. jQuery最重要的知识点

    1.各种常见的选择器.2.对于属性的操作.[重点] 2.1)获取或设置属性的值: prop(); 2.2 ) 添加.删除.切换样式: addClass/removeClass/toggleClass ...

  5. ctf题目writeup(1)

    2019/1/28 题目来源:爱春秋 https://www.ichunqiu.com/battalion?t=1 1. 该文件是一个音频文件: 首先打开听了一下,有短促的长的....刚开始以为是摩斯 ...

  6. go 操作数据库

    假设有了数据库,创建表 CREATE TABLE `userinfo` ( `uid` INT(10) NOT NULL AUTO_INCREMENT, //自增字段 `username` VARCH ...

  7. SpringCloud项目,接口调用返回http 500 - Internal Server Error的错误

    今天上班的时候,自己正在参与的Spring Cloud项目出现了问题,原本上周五还正常的项目突然所有接口调用都是返回http 500的错误. 项目的状态是在Eureka上可以看到对应微服务是在线状态, ...

  8. APIO2018 游记

    day \(-\infty\) \(\sim\) day0 5 月 5 号左右的时候去了趟中北大学,山西省大学生程序设计竞赛.不是太满意,现场 rk3.拿到了充电宝(冲着这个去的,虽然抵不过车费),抽 ...

  9. Eclipse AmaterasUML 安装及使用

    AmaterasUML 对于我来说,是一个非常好用的UML插件. 用它来将我写过的一些Android程序进行逆工程非常好用,只不过,不能体现出包,这是一个小小的遗憾. 这个是它的主页地址:http:/ ...

  10. HDU 3697 Selecting courses(贪心+暴力)(2010 Asia Fuzhou Regional Contest)

    Description     A new Semester is coming and students are troubling for selecting courses. Students ...