2090. 「ZJOI2016」旅行者

链接

loj

思路

\((l,mid)(mid+1,r)\).考虑跨过mid的贡献。

假设选的中间那条线的点为gzy,贡献为\(dis(x,gzy)+dis(gzy,y)\)

那就计算n遍最短路,一次分治为\(n^2mlog{nm}\)

设S=n*m.矩阵的长度是不定的,每次取最长的边进行分治是最好的,n最坏为\(\sqrt{n}\)。

\(f(n)=2*f(\frac{n}{2})+S\sqrt{S}logS。所以总的复杂度就是\)\(S\sqrt{S}logS\)

都在同侧的也需要跨一跨

代码

#include <bits/stdc++.h>
#define FOR(i,a,b) for(int i=a;i<=b;++i)
using namespace std;
const int _=1e5+7,INF=0x3f3f3f3f;
int read() {
int x=0,f=1;char s=getchar();
for(;s>'9'||s<'0';s=getchar()) if(s=='-') f=-1;
for(;s>='0'&&s<='9';s=getchar()) x=x*10+s-'0';
return x*f;
}
int n,m,q,ans[_],vis[_];
struct node {
int x,y,X,Y,u,v,id;
}Q[_],tmp[_];
bool cmp(node a,node b) {return a.id<b.id;}
struct edge {int v,nxt,q;}e[_];
int head[_],tot;
void add(int u,int v,int q) {
e[++tot].v=v;
e[tot].q=q;
e[tot].nxt=head[u];
head[u]=tot;
}
int id(int x,int y) {return (x-1)*m+y;}
struct T_T {
int u,val;
T_T(int a=0,int b=0) {u=a,val=b;}
bool operator < (const T_T &b) const {return val>b.val;}
};
int dis[_];
void dij(int S) {
dis[S]=0;
priority_queue<T_T> q;
q.push(T_T(S,0));
while(!q.empty()) {
T_T u=q.top();q.pop();
if(dis[u.u]!=u.val) continue;
for(int i=head[u.u];i;i=e[i].nxt) {
int v=e[i].v;
if(vis[v]&&dis[v]>u.val+e[i].q) {
dis[v]=u.val+e[i].q;
q.push(T_T(v,dis[v]));
}
}
}
}
void solve(int x,int y,int X,int Y,int l,int r) {
if(l>r) return;
if(x==X&&y==Y) {
for(int i=l;i<=r;++i) ans[Q[i].id]=0;
return;
}
if(Y-y>X-x) {
int mid=(Y+y)>>1;
FOR(i,x,X) {
FOR(j,x,X) FOR(k,y,Y) dis[id(j,k)]=INF;
dij(id(i,mid));
FOR(j,l,r)
ans[Q[j].id]=min(dis[Q[j].u]+dis[Q[j].v],ans[Q[j].id]);
}
FOR(i,x,X) vis[id(i,mid)]=0;
int p=l,q=r;
FOR(i,l,r) {
if(max(Q[i].y,Q[i].Y)<=mid) tmp[p++]=Q[i];
if(min(Q[i].y,Q[i].Y)>mid) tmp[q--]=Q[i];
}
FOR(i,l,r) Q[i]=tmp[i];
solve(x,y,X,mid,l,p-1);
solve(x,mid+1,X,Y,q+1,r);
} else {
int mid=(X+x)>>1;
FOR(i,y,Y) {
FOR(j,x,X) FOR(k,y,Y) dis[id(j,k)]=INF;
dij(id(mid,i));
FOR(j,l,r)
ans[Q[j].id]=min(dis[Q[j].u]+dis[Q[j].v],ans[Q[j].id]);
}
FOR(i,y,Y) vis[id(mid,i)]=0;
int p=l,q=r;
FOR(i,l,r) {
if(max(Q[i].x,Q[i].X)<=mid) tmp[p++]=Q[i];
if(min(Q[i].x,Q[i].X)>mid) tmp[q--]=Q[i];
}
FOR(i,l,r) Q[i]=tmp[i];
solve(x,y,mid,Y,l,p-1);
solve(mid+1,y,X,Y,q+1,r);
}
}
int main() {
n=read(),m=read();
FOR(i,1,n) FOR(j,1,m-1) {
int val=read();
add(id(i,j),id(i,j+1),val);
add(id(i,j+1),id(i,j),val);
}
FOR(i,1,n-1) FOR(j,1,m) {
int val=read();
add(id(i,j),id(i+1,j),val);
add(id(i+1,j),id(i,j),val);
}
FOR(i,1,n) FOR(j,1,m) vis[id(i,j)]=1;
q=read();
FOR(i,1,q) {
Q[i].x=read(),Q[i].y=read(),
Q[i].u=id(Q[i].x,Q[i].y);
Q[i].X=read(),Q[i].Y=read(),
Q[i].v=id(Q[i].X,Q[i].Y);
Q[i].id=i,ans[i]=INF;
}
solve(1,1,n,m,1,q);
FOR(i,1,q) printf("%d\n",ans[i]);
return 0;
}

2090. 「ZJOI2016」旅行者 分治,最短路的更多相关文章

  1. @loj - 2090@ 「ZJOI2016」旅行者

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 小 Y 来到了一个新的城市旅行.她发现了这个城市的布局是网格状的 ...

  2. 【LOJ】#2090. 「ZJOI2016」旅行者

    题解 每次按较长边把矩形分成两半,找一个中间轴,轴上的每个点跑一边最短路更新所有的答案 然后把矩形分成两半,递归下去 代码 #include <bits/stdc++.h> #define ...

  3. 「ZJOI2016」旅行者 解题报告

    「ZJOI2016」旅行者 对网格图进行分治. 每次从中间选一列,然后枚举每个这一列的格子作为起点跑最短路,进入子矩形时把询问划分一下,有点类似整体二分 至于复杂度么,我不会阿 Code: #incl ...

  4. loj2090 「ZJOI2016」旅行者

    分治+最短路,很套路的 #include <algorithm> #include <iostream> #include <cstring> #include & ...

  5. 「ZJOI2016」解题报告

    「ZJOI2016」解题报告 我大浙的省选题真是超级神仙--这套已经算是比较可做的了. 「ZJOI2016」旅行者 神仙分治题. 对于一个矩形,每次我们从最长边切开,最短边不会超过 \(\sqrt{n ...

  6. 「ZJOI2016」大森林 解题报告

    「ZJOI2016」大森林 神仙题... 很显然线段树搞不了 考虑离线操作 我们只搞一颗树,从位置1一直往后移动,然后维护它的形态试试 显然操作0,1都可以拆成差分的形式,就是加入和删除 因为保证了操 ...

  7. 「bzoj1003」「ZJOI2006」物流运输 最短路+区间dp

    「bzoj1003」「ZJOI2006」物流运输---------------------------------------------------------------------------- ...

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

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

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

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

随机推荐

  1. Python 遍历目录下的子目录和文件

    import os A: 遍历目录下的子目录和文件 for root,dirs ,files in os.walk(path) root:要访问的路径名 dirs:遍历目录下的子目录 files:遍历 ...

  2. Dubbo简介与使用

    1:Dubbo是什么 阿里生产的一种rpc 实现框架  Dubbo 是一个分布式服务框架,是阿里巴巴开源项目 ,被国内电商及互联网项目中使用. Dubbo 致力于提供高性能和透明化的RPC远程服务调用 ...

  3. FZU 1759 题解 欧拉降幂

    本题考点:欧拉降幂 Super A^B mod C Given A,B,C, You should quickly calculate the result of A^B mod C. (1<= ...

  4. why’s kafka so fast

    As we all know that Kafka is very fast, much faster than most of its competitors. So what’s the reas ...

  5. Android 支持库迁移到AndroidX

    一.背景 Android系统版本在不断更新,从最初的Android 1.0到现在Google和各大手机厂商正在推的Android 10,平均下来每个年头都有一个大的版本更新.但用户正在用的手机上的An ...

  6. NFS客户端挂载失败之authenticated unmount request from

    1.故障现象 客户端挂载时夯住,无反应,无报错,如下图: 2.故障前对挂载目录的操作 发现故障前挂载目录被误删除,后通过备份分为恢复 3.故障排查步骤 .检查客户端及服务端防火墙规则 .检查selin ...

  7. spring-session(一)揭秘续篇

    上一篇文章中介绍了Spring-Session的核心原理,Filter,Session,Repository等等,传送门:spring-session(一)揭秘. 这篇继上一篇的原理逐渐深入Sprin ...

  8. 整理:C#常用字符串操作,常用数值类型转换

    一.字符串操作 1. 字符串连接 //将指定的数组所有元素拼接为一个字符串 string[] arr = {"A","B","C"}; st ...

  9. 在VB编程中,若一行代码太长需要换行时,行尾要加什么符号

    & _ 注意,&与_之间一定要有一个空格 例如: aa="select " & _     "  a,b,c" & _      ...

  10. H5页面基础元素

    H5页面结构元素示例 <!DOCTYPE html> <html lang="zh"> <head> <meta charset=&quo ...