loj2090 「ZJOI2016」旅行者
分治+最短路,很套路的
#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
#include <queue>
using namespace std;
int n, m, ans[100005], px[20005], py[20005], uu, hea[20005], cnt, q, din;
int dis[20005];
bool vis[20005];
struct Edge{
int too, nxt, val;
}edge[1000005];
struct Ques{
int xu, yu, xv, yv, id;
}qu[100005], qq[100005];
struct Node{
int idx, dis;
}nd[5000005];
int f(int x, int y){
return (x-1)*m+y;
}
void add_edge(int fro, int too, int val){
edge[++cnt].nxt = hea[fro];
edge[cnt].too = too;
edge[cnt].val = val;
hea[fro] = cnt;
}
bool inn(int x, int nl, int nr, int ml, int mr){
if(px[x]>=nl && px[x]<=nr && py[x]>=ml && py[x]<=mr)
return true;
return false;
}
bool cmp(Node x, Node y){
return x.dis>y.dis;
}
void dijkstra(int x, int nl, int nr, int ml, int mr){
for(int i=nl; i<=nr; i++)
for(int j=ml; j<=mr; j++){
dis[f(i,j)] = 0x3f3f3f3f;
vis[f(i,j)] = false;
}
dis[x] = 0;
din = 1;
nd[1] = (Node){x, 0};
while(din){
int o=nd[1].idx;
pop_heap(nd+1, nd+1+din, cmp);
din--;
if(vis[o]) continue;
vis[o] = true;
for(int i=hea[o]; i; i=edge[i].nxt){
int t=edge[i].too;
if(inn(t, nl, nr, ml, mr) && dis[t]>dis[o]+edge[i].val){
dis[t] = dis[o] + edge[i].val;
nd[++din] = (Node){t, dis[t]};
push_heap(nd+1, nd+1+din, cmp);
}
}
}
}
void solve(int nl, int nr, int ml, int mr, int ql, int qr){
if(ql>qr) return ;
if(nr-nl>mr-ml){
int mid=(nl+nr)>>1;
for(int i=ml; i<=mr; i++){
int p=f(mid, i);
dijkstra(p, nl, nr, ml, mr);
for(int j=ql; j<=qr; j++)
ans[qu[j].id] = min(ans[qu[j].id], dis[f(qu[j].xu, qu[j].yu)]+dis[f(qu[j].xv, qu[j].yv)]);
}
int l=ql, r=qr;
for(int i=ql; i<=qr; i++)
if(qu[i].xu<mid && qu[i].xv<mid) qq[l++] = qu[i];
else if(qu[i].xu>mid && qu[i].xv>mid) qq[r--] = qu[i];
for(int i=ql; i<=qr; i++)
qu[i] = qq[i];
solve(nl, mid-1, ml, mr, ql, l-1);
solve(mid+1, nr, ml, mr, r+1, qr);
}
else{
int mid=(ml+mr)>>1;
for(int i=nl; i<=nr; i++){
int p=f(i, mid);
dijkstra(p, nl, nr, ml, mr);
for(int j=ql; j<=qr; j++)
ans[qu[j].id] = min(ans[qu[j].id], dis[f(qu[j].xu, qu[j].yu)]+dis[f(qu[j].xv, qu[j].yv)]);
}
int l=ql, r=qr;
for(int i=ql; i<=qr; i++)
if(qu[i].yu<mid && qu[i].yv<mid) qq[l++] = qu[i];
else if(qu[i].yu>mid && qu[i].yv>mid) qq[r--] = qu[i];
for(int i=ql; i<=qr; i++)
qu[i] = qq[i];
solve(nl, nr, ml, mid-1, ql, l-1);
solve(nl, nr, mid+1, mr, r+1, qr);
}
}
int main(){
memset(ans, 0x3f, sizeof(ans));
cin>>n>>m;
for(int i=1; i<=n; i++)
for(int j=1; j<m; j++){
scanf("%d", &uu);
int p=f(i,j), q=f(i,j+1);
add_edge(p, q, uu);
add_edge(q, p, uu);
}
for(int i=1; i<n; i++)
for(int j=1; j<=m; j++){
scanf("%d", &uu);
int p=f(i,j), q=f(i+1,j);
add_edge(p, q, uu);
add_edge(q, p, uu);
}
for(int i=1; i<=n; i++)
for(int j=1; j<=m; j++){
int p=f(i,j);
px[p] = i;
py[p] = j;
}
cin>>q;
for(int i=1; i<=q; i++){
scanf("%d %d %d %d", &qu[i].xu, &qu[i].yu, &qu[i].xv, &qu[i].yv);
qu[i].id = i;
}
solve(1, n, 1, m, 1, q);
for(int i=1; i<=q; i++)
printf("%d\n", ans[i]);
return 0;
}
loj2090 「ZJOI2016」旅行者的更多相关文章
- 「ZJOI2016」旅行者 解题报告
「ZJOI2016」旅行者 对网格图进行分治. 每次从中间选一列,然后枚举每个这一列的格子作为起点跑最短路,进入子矩形时把询问划分一下,有点类似整体二分 至于复杂度么,我不会阿 Code: #incl ...
- 2090. 「ZJOI2016」旅行者 分治,最短路
2090. 「ZJOI2016」旅行者 链接 loj 思路 \((l,mid)(mid+1,r)\).考虑跨过mid的贡献. 假设选的中间那条线的点为gzy,贡献为\(dis(x,gzy)+dis(g ...
- 【LOJ】#2090. 「ZJOI2016」旅行者
题解 每次按较长边把矩形分成两半,找一个中间轴,轴上的每个点跑一边最短路更新所有的答案 然后把矩形分成两半,递归下去 代码 #include <bits/stdc++.h> #define ...
- @loj - 2090@ 「ZJOI2016」旅行者
目录 @description@ @solution@ @accepted code@ @details@ @description@ 小 Y 来到了一个新的城市旅行.她发现了这个城市的布局是网格状的 ...
- 「ZJOI2016」解题报告
「ZJOI2016」解题报告 我大浙的省选题真是超级神仙--这套已经算是比较可做的了. 「ZJOI2016」旅行者 神仙分治题. 对于一个矩形,每次我们从最长边切开,最短边不会超过 \(\sqrt{n ...
- 「ZJOI2016」大森林 解题报告
「ZJOI2016」大森林 神仙题... 很显然线段树搞不了 考虑离线操作 我们只搞一颗树,从位置1一直往后移动,然后维护它的形态试试 显然操作0,1都可以拆成差分的形式,就是加入和删除 因为保证了操 ...
- loj2092 「ZJOI2016」大森林
ref不是太懂-- #include <algorithm> #include <iostream> #include <cstring> #include < ...
- loj2091 「ZJOI2016」小星星
ref 总的来说,就是 容斥转化为点对应到点集问题. 树形 dp 解决转化后的问题. #include <iostream> #include <cstring> #inclu ...
- 「ZJOI2016」小星星
传送门 Description Solution 容斥,考虑有多少个节点不被匹配到,求出的方案,多个点可以同时不被匹配到 状态压缩+树形dp Code #include<bits/stdc++ ...
随机推荐
- javascript的常用操作(一)
1. 实时监听input的值变化 onchange事件只在键盘或者鼠标操作改变对象属性,且失去焦点时触发,脚本触发无效; 而onkeydown/onkeypress/onkeyup在处理复制.粘贴. ...
- wechat开发笔记之1.接口示例代码
修改后的php示例代码! <?php /** * wechat php test */ //define your token define("TOKEN", "w ...
- php-7.1.11编译选项配置
./configure \ --prefix=/usr/local/php-7.1.11 \ --with-config-file-path=/usr/local/php7.1.11/etc \ -- ...
- IIS7.5配置自动添加www 及 限制通过IP访问web
IIS7.5配置自动添加www 方法 新建一个站点2(主机名为不带www的站点),将其重定向至带www的URL即可. 注意以下几点 站点2不可与站点1的路径一致,否则会导致站点1也会添加同样的重定向, ...
- Linux I/O调度
一) I/O调度程序的总结 1) 当向设备写入数据块或是从设备读出数据块时,请求都被安置在一个队列中等待完成. 2) 每个块设备都有它自己的队列. 3) I/O调度程序负责维护这些队列的顺 ...
- Linux环境下使用xampp配置php开发环境
XAMPP (Apache+MySQL+PHP+PERL)是一个功能强大的建站集成软件包.这个软件包原来的名字是LAMPP,但是为 了避免误 解,最新的几个版本就改名为 XAMPP 了.它可以在Win ...
- 在 Java 8 中避免 Null 检查
如何预防 Java 中著名的 NullPointerException 异常?这是每个 Java 初学者迟早会问到的关键问题之一.而且中级和高级程序员也在时时刻刻规避这个错误.其是迄今为止 Java ...
- JavaScript 常用的排序算法
冒泡排序 function bubbleSort(array) { for (let i = 0; i < array.length; i++) for (let j = 0; j < a ...
- 数据结构(C语言)分享笔记:数据结构的逻辑层次、存储层次
[1] 严格意义上数据结构的概念 数据结构,一个简单的定义:相互之间存在一种或多种特定关系的数据元素的集合.即:数据结构 = 元素集合 + 元素间关系的集合 . 在讨论数据结构时,可以基于两个不同的层 ...
- Spring Boot 前世今生
Spring Boot 2.0 的推出又激起了一阵学习 Spring Boot 热,就单从我个人的博客的访问量大幅增加就可以感受到大家对学习 Spring Boot 的热情,那么在这么多人热衷于学习 ...