Description

小Y来到了一个新的城市旅行。她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北
的道路,这些道路两两相交形成n×m个路口 (i,j)(1≤i≤n,1≤j≤m)。她发现不同的道路路况不同,所以通过不
同的路口需要不同的时间。通过调查发现,从路口(i,j)到路口(i,j+1)需要时间 r(i,j),从路口(i,j)到路口(i+1
,j)需要时间c(i,j)。注意这里的道路是双向的。小Y有q个询问,她想知道从路口(x1,y1)到路口(x2,y2)最少需要
花多少时间。

Input

第一行包含 2 个正整数n,m,表示城市的大小。
 
接下来n行,每行包含m?1个整数,第i行第j个正整数表示从一个路口到另一个路口的时间r(i,j)。
 
接下来n?1行,每行包含m个整数,第i行第j个正整数表示从一个路口到另一个路口的时间c(i,j)。
 
接下来一行,包含1个正整数q,表示小Y的询问个数。
 
接下来q行,每行包含4个正整数 x1,y1,x2,y2,表示两个路口的位置。

Output

输出共q行,每行包含一个整数表示从一个路口到另一个路口最少需要花的时间。

Sample Input

2 2
2
3
6 4
2
1 1 2 2
1 2 2 1

Sample Output

6
7

Solution

网格图求任意两点间的最短路。

可以用分治来解决。

之前校内训练的时候CJK学长出了一道IOI2013的题,就是用线段树来维护网格图的最短路。这题也很类似,离线询问以后,每次把长边拿出来分治,考虑经过中间这一排点的和没经过这一排点的。没经过的递归下去做,经过的就跑一遍堆优化dj或者spfa就好了。

Code

 #include <cstdio>
#include <cstring> #define R register
#define maxn 20010
#define cmin(_a, _b) (_a > (_b) ? _a = (_b) : 0)
#define id(_a, _b) (((_a) - 1) * m + (_b) - 1)
#define id1(_x) ((_x) / m + 1)
#define id2(_x) ((_x) % m + 1)
int n, m;
struct Edge {
Edge *next;
int to, w;
} *last[maxn], e[maxn << ], *ecnt = e;
inline void link(R int a, R int b, R int w)
{
*++ecnt = (Edge) {last[a], b, w}; last[a] = ecnt;
*++ecnt = (Edge) {last[b], a, w}; last[b] = ecnt;
}
struct Ques {
int x1, y1, x2, y2, id;
} qu[], tmp[];
int ans[], dis[maxn], q[maxn * ], r[maxn], c[maxn];
bool inq[maxn];
#define inf 0x7fffffff
struct Data {
int pos, dis;
inline bool operator < (const Data &that) const {return dis > that.dis;}
} ;
#include <queue>
std::priority_queue<Data> hp;
void spfa(R int s, R int nl, R int nr, R int ml, R int mr)
{
//for (R int i = nl; i <= nr; ++i) for (R int j = ml; j <= mr; ++j) dis[id(i, j)] = inf;
/*
R int head = maxn * 20, tail = maxn * 20 + 1;
q[maxn * 20 + 1] = s; dis[s] = 0;
*/
hp.push((Data) {s, dis[s] = });
while (/*head < tail*/!hp.empty())
{
// R int now = q[++head]; inq[now] = 0;
R Data tp = hp.top(); hp.pop();
R int now = tp.pos;
for (R Edge *iter = last[now]; iter; iter = iter -> next)
if (dis[iter -> to] > dis[now] + iter -> w && nl <= id1(iter -> to) && id1(iter -> to) <= nr && ml <= id2(iter -> to) && id2(iter -> to) <= mr)
{
dis[iter -> to] = dis[now] + iter -> w;
// !inq[iter -> to] ? inq[dis[iter -> to] < dis[q[head + 1]] ? q[head--] = iter -> to : q[++tail] = iter -> to] = 1 : 0;
hp.push((Data) {iter -> to, dis[iter -> to]});
}
}
}
void work(R int nl, R int nr, R int ml, R int mr, R int ql, R int qr)
{
if (nl > nr || ml > mr) return ;
if (ql > qr) return ;
if (nr - nl + <= mr - ml + )
{
R int mid = ml + mr >> ;
for (R int i = nl; i <= nr; ++i) for (R int j = ml; j <= mr; ++j) dis[id(i, j)] = inf;
for (R int i = nl; i <= nr; ++i)
{
if (i != nl)
{
for (R int ii = nl; ii <= nr; ++ii) for (R int jj = ml; jj <= mr; ++jj)
dis[id(ii, jj)] += c[id(i - , mid)];
}
spfa(id(i, mid), nl, nr, ml, mr);
for (R int j = ql; j <= qr; ++j)
cmin(ans[qu[j].id], dis[id(qu[j].x1, qu[j].y1)] + dis[id(qu[j].x2, qu[j].y2)]);
}
R int qql = ql - , qqr = qr + ;
for (R int i = ql; i <= qr; ++i)
if (qu[i].y1 < mid && qu[i].y2 < mid)
tmp[++qql] = qu[i];
else if (qu[i].y1 > mid && qu[i].y2 > mid)
tmp[--qqr] = qu[i]; for (R int i = ql; i <= qql; ++i) qu[i] = tmp[i];
for (R int i = qqr; i <= qr; ++i) qu[i] = tmp[i];
work(nl, nr, ml, mid - , ql, qql);
work(nl, nr, mid + , mr, qqr, qr);
}
else
{
R int mid = nl + nr >> ;
for (R int i = nl; i <= nr; ++i) for (R int j = ml; j <= mr; ++j) dis[id(i, j)] = inf;
for (R int i = ml; i <= mr; ++i)
{
if (i != ml)
{
for (R int ii = nl; ii <= nr; ++ii) for (R int jj = ml; jj <= mr; ++jj)
dis[id(ii, jj)] += r[id(mid, i - )];
}
spfa(id(mid, i), nl, nr, ml, mr);
for (R int j = ql; j <= qr; ++j)
cmin(ans[qu[j].id], dis[id(qu[j].x1, qu[j].y1)] + dis[id(qu[j].x2, qu[j].y2)]);
}
R int qql = ql - , qqr = qr + ;
for (R int i = ql; i <= qr; ++i)
if (qu[i].x1 < mid && qu[i].x2 < mid)
tmp[++qql] = qu[i];
else if (qu[i].x1 > mid && qu[i].x2 > mid)
tmp[--qqr] = qu[i]; for (R int i = ql; i <= qql; ++i) qu[i] = tmp[i];
for (R int i = qqr; i <= qr; ++i) qu[i] = tmp[i];
work(nl, mid - , ml, mr, ql, qql);
work(mid + , nr, ml, mr, qqr, qr);
}
}
int main()
{
scanf("%d%d", &n, &m);
for (R int i = ; i <= n; ++i) for (R int j = ; j < m; ++j)
{R int w; scanf("%d", &w); link(id(i, j), id(i, j + ), w); r[id(i, j)] = w;}
for (R int i = ; i < n; ++i) for (R int j = ; j <= m; ++j)
{R int w; scanf("%d", &w); link(id(i, j), id(i + , j), w); c[id(i, j)] = w;}
R int Q; scanf("%d", &Q);
for (R int i = ; i <= Q; ++i) scanf("%d%d%d%d", &qu[i].x1, &qu[i].y1, &qu[i].x2, &qu[i].y2), qu[i].id = i;
memset(ans, , (Q + ) << );
work(, n, , m, , Q);
for (R int i = ; i <= Q; ++i) printf("%d\n", ans[i]);
return ;
}

【BZOJ4456】 [Zjoi2016]旅行者 / 【UOJ #184】 【ZJOI2016】旅行者的更多相关文章

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

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

  2. 【BZOJ 4456】【UOJ #184】【ZJOI 2016】旅行者

    http://www.lydsy.com/JudgeOnline/problem.php?id=4456 http://uoj.ac/problem/184 参考(抄)的晨爷的题解(代码) 对矩形进行 ...

  3. [CNBETA]动图告诉你 光速到底有多慢?

    https://www.cnbeta.com/articles/tech/811381.htm 我们知道,30万公里每秒的光速是宇宙内目前已知的最高速度,至少现有人类理论体系下它是不可跨越的.30万公 ...

  4. bzoj4456: [Zjoi2016]旅行者

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

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

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

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

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

  7. BZOJ4456/UOJ184 [Zjoi2016]旅行者

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...

  8. BZOJ4456 ZJOI2016旅行者(分治+最短路)

    感觉比较套路,每次在长边中轴线处切一刀,求出切割线上的点对矩形内所有点的单源最短路径,以此更新每个询问,递归处理更小的矩形.因为若起点终点跨过中轴线是肯定要经过的,而不跨过中轴线的则可以选择是否经过中 ...

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

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

随机推荐

  1. .Net Core 中使用NLog作为日志中间件

    ⒈安装相关依赖 NLog NLog.Web.AspNetCore ⒉在项目的根目录中创建NLog配置文件 <?xml version="1.0" encoding=" ...

  2. C++11智能指针原理和实现

    一.智能指针起因 在C++中,动态内存的管理是由程序员自己申请和释放的,用一对运算符完成:new和delete. new:在动态内存中为对象分配一块空间并返回一个指向该对象的指针: delete:指向 ...

  3. C# 连接 Oracle数据库增删改查,事务

    一. 前情提要 一般.NET环境连接Oracle数据库,是通过 TNS/SQL.NET 配置文件,而 TNS 必须要 Oracle 客户端(如果连接的是服务器的数据库,本地还要装一个 client , ...

  4. Flask-migrate基本使用方法

    数据库迁移操作顺序: 1.python 文件 db init 2.根据需求修改模型 3.python flaskapp文件 db migrate -m"新版本名(注释)" 4.py ...

  5. Java开发者想尝试转行大数据,学习方向建议?

      ​前言 相信很多Java开发者都对大数据有一定的了解,随着大数据时代的到来,也有很多Java程序员想要转行大数据.大数据技术中大多数平台使用的都是Java语言,因此,对于大数据技术的学习来说,Ja ...

  6. 06 Nginx

    1.检查linux上是否通过yum安装了nginx rpm -qi nginx 2.解决安装nginx所依赖包 yum install gcc patch libffi-devel python-de ...

  7. Nginx安装目录详解

    Nginx安装目录详解 1. 查看有关nginx的所有目录列表,输入命令  rpm -ql nginx 可以查看有关nginx目录信息,但是注意 这种命令只能是在基于yum安装的方式才可以. 2. 下 ...

  8. Mysql检查列是否存在并新增、修改、删除列

    在MYSQL中,新增.修改.删除列时不能进行IF EXISTS判断,IF语句只能出现在存储过程当中,故博主用存储过程的方法新增.修改.删除列,修改列名称. DROP PROCEDURE IF EXIS ...

  9. 第十篇.2、python并发编程之多进程

    一 multiprocessing模块介绍 python中的多线程无法利用多核优势,如果想要充分地使用多核CPU的资源(os.cpu_count()查看),在python中大部分情况需要使用多进程.P ...

  10. 升级python导致yum报错的解决方法

    把python从2.7升级到3.6后 , 使用yum报错 File ‘’/usr/bin/yum'', line 30 except KeyboardInterrupt, e: ^ 故障原因:yum采 ...