4456: [Zjoi2016]旅行者

https://www.lydsy.com/JudgeOnline/problem.php?id=4456

分析:

  每次对当前矩阵按长边化一条分治线,然后在对分治线上的点跑最短路,然后可以处理处过分治线的询问。对于不过分治线的,递归处理。

  先写的dijkstra+堆优化,在开O2的情况下可以过,不开O2过不了,卡常~,还是过不了。然后在UOJ的排行榜里(而且UOJ是有大样例的!),看到了两个优化,加上就可以了。

  优化:1、代码41行,用上次的遍历结果初始化。2、将dijkstra改成spfa+SLF优化。

代码:

 #include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<iostream>
using namespace std;
typedef long long LL; char buf[],*_p1 = buf,*_p2 = buf;
#define nc() (_p1==_p2&&(_p2=(_p1=buf)+fread(buf,1,100000,stdin),_p1==_p2) ? EOF :*_p1++)
inline int read() {
int x=,f=;char ch=nc();for(;!isdigit(ch);ch=nc())if(ch=='-')f=-;
for (;isdigit(ch);ch=nc())x=x*+ch-'';return x*f;
} #define getid(a,b) ((a - 1) * m + b)
#define pa pair<int,int>
#define mp(a,b) make_pair(a,b) const int INF = 1e9;
const int N = ; struct Que{
int a1,b1,a2,b2,id;
}A[N], B[N];
int dis[N], ans[N], head[N], nxt[N], to[N], len[N];
int n, m, En;
//priority_queue< pa, vector< pa >, greater< pa > > q; // 这才是小根堆!!!
int q[];
bool vis[N]; inline void getxy(int id,int &x,int &y) {
x = (id - ) / m + ;
y = (id - ) % m + ;
}
void dijkstra(int s,int a1,int a2,int b1,int b2,int h) {
int x, y, d = dis[s];
for (int i=a1; i<=a2; ++i)
for (int j=b1; j<=b2; ++j) {
x = getid(i, j);
dis[x] = h ? dis[x] + d : INF;
vis[x] = false;
}
dis[s] = ;
/* q.push(mp(dis[s], s));
while (!q.empty()) {
pa now = q.top(); q.pop();
int u = now.second;
if (vis[u]) continue;
vis[u] = true;
for (int i=head[u]; i; i=nxt[i]) {
int v = to[i];
getxy(v, x, y);
if (x >= a1 && x <= a2 && y >= b1 && y <= b2 && dis[v] > dis[u] + len[i]) {
dis[v] = dis[u] + len[i];
q.push(mp(dis[v], v));
}
}
}*/
int L = , R = L - ;
q[++R] = s;vis[s] = true;
while (L <= R) {
int u = q[L ++];vis[u] = false;
for (int i=head[u]; i; i=nxt[i]) {
int v = to[i];
getxy(v, x, y);
if (x >= a1 && x <= a2 && y >= b1 && y <= b2 && dis[v] > dis[u] + len[i]) {
dis[v] = dis[u] + len[i];
if (!vis[v]) {
if (dis[v] <= dis[q[L]]) q[--L] = v;
else q[++R] = v;
vis[v] = true;
}
}
}
}
}
void solve(int a1,int a2,int b1,int b2,int L,int R) {
if (L > R) return ;
int i, j, k;
if (a2 - a1 > b2 - b1) {
int mid = (a2 + a1) / ; // a2 + a1
for (i=b1; i<=b2; ++i) {
dijkstra(getid(mid, i), a1, a2, b1, b2, i-b1);
for (j=L; j<=R; ++j)
ans[A[j].id] = min(ans[A[j].id], dis[getid(A[j].a1, A[j].b1)] + dis[getid(A[j].a2, A[j].b2)]);
}
i = L - , j = R + ;
for (k=L; k<=R; ++k) {
if (A[k].a1 < mid && A[k].a2 < mid) B[++i] = A[k];
if (A[k].a1 > mid && A[k].a2 > mid) B[--j] = A[k];
}
for (k=L; k<=i; ++k) A[k] = B[k]; solve(a1, mid-, b1, b2, L, i);
for (k=j; k<=R; ++k) A[k] = B[k]; solve(mid+, a2, b1, b2, j, R);
}
else {
int mid = (b2 + b1) / ;
for (i=a1; i<=a2; ++i) {
dijkstra(getid(i, mid), a1, a2, b1, b2, i-a1);
for (j=L; j<=R; ++j)
ans[A[j].id] = min(ans[A[j].id], dis[getid(A[j].a1, A[j].b1)] + dis[getid(A[j].a2, A[j].b2)]);
}
i = L - , j = R + ;
for (k=L; k<=R; ++k) {
if (A[k].b1 < mid && A[k].b2 < mid) B[++i] = A[k];
if (A[k].b1 > mid && A[k].b2 > mid) B[--j] = A[k];
}
for (k=L; k<=i; ++k) A[k] = B[k]; solve(a1, a2, b1, mid-, L, i);
for (k=j; k<=R; ++k) A[k] = B[k]; solve(a1, a2, mid+, b2, j, R);
}
}
inline void add_edge(int u,int v,int w) {
++En; to[En] = v; len[En] = w; nxt[En] = head[u]; head[u] = En;
++En; to[En] = u; len[En] = w; nxt[En] = head[v]; head[v] = En;
}
int main() {
n = read(), m = read();
for (int i=; i<=n; ++i) {
for (int j=; j<m; ++j) {
int a = read();
add_edge(getid(i, j), getid(i, j + ), a);
}
}
for (int i=; i<n; ++i) {
for (int j=; j<=m; ++j) {
int a = read();
add_edge(getid(i, j), getid(i + , j), a);
}
}
int Case = read();
for (int a,b,i=; i<=Case; ++i) {
A[i].a1 = read(), A[i].b1 = read();
A[i].a2 = read(), A[i].b2 = read();
A[i].id = i; ans[i] = INF;
}
solve(, n, , m, , Case);
for (int i=; i<=Case; ++i) printf("%d\n",ans[i]);
return ;
}

4456: [Zjoi2016]旅行者的更多相关文章

  1. ●BOZJ 4456 [Zjoi2016]旅行者

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=4456 题解: 分治好题.大致做法如下:对于一开始的矩形区域,过较长边的中点把矩形区域分为两个 ...

  2. bzoj 4456 [Zjoi2016]旅行者

    题面 https://www.lydsy.com/JudgeOnline/problem.php?id=4456 题解 分治 设当前work的区间为(x1,y1,x2,y2) 我们将长边分成两半 不妨 ...

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

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

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

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

  5. P3350 [ZJOI2016]旅行者

    题目描述 小Y来到了一个新的城市旅行.她发现了这个城市的布局是网格状的,也就是有n条从东到西的道路和m条从南到北的道路,这些道路两两相交形成n*m个路口 (i,j)(1<=i<=n,1&l ...

  6. bzoj4456: [Zjoi2016]旅行者

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

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

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

  8. luogu3350 [ZJOI2016]旅行者

    链接 P3350 [ZJOI2016]旅行者 题目大意:给出网格图,求两点之间最短路,多组询问. \(n*m\leq10^5\ \ q\leq 10^5\) 考虑\(CDQ\)分治. 首先把询问离线, ...

  9. Luogu 3350 [ZJOI2016]旅行者

    BZOJ 4456 听若干个大佬讲过$n$遍终于写掉了. 我把时限基本上跑满了2333…… 分治 + 最短路. 首先我们去分治这个矩形格子,找到一条长边把它对半切,对切开的边上的每一个点跑一遍最短路然 ...

随机推荐

  1. 四、HTTP和HTTPS的区别

    1:简介 超文本传输协议HTTP协议被用于在Web浏览器和网站服务器之间传递信息.HTTP协议以明文方式发送内容,不提供任何方式的数据加密,如果攻击者截取了Web浏览器和网站服务器之间的传输报文,就可 ...

  2. PhoneGap 的存储 API_Web Sql

    一.介绍 此 API 基于 W3C WEB SQL Database Specification 和 W3C Web Storage API Specification. 有些 设备已经提供了对该规范 ...

  3. BIND简易教程(0):在Ubuntu下源码安装BIND(其实跟前面的教程没太大关系)

    之前介绍过BIND的基本使用啦.关于BIND的入门级使用方法见:http://www.cnblogs.com/anpengapple/p/5877661.html简易教程系列,本篇只讲BIND安装. ...

  4. springMvc返回Json中自定义日期格式

    (一)输出json数据 springmvc中使用jackson-mapper-asl即可进行json输出,在配置上有几点: 1.使用mvc:annotation-driven 2.在依赖管理中添加ja ...

  5. Yii 验证和消息

    setFlash(), getFlash()可以完成验证成功后提示 <?php # 成功信息提示 Yii::app()->user->setFlash('success', &quo ...

  6. CTSC2018 && APIO2018 && SDOI2018R2游记

    Day -? 占个坑先.希望CTSC,APIO别打铁,R2别滚粗QAQ CTSC Day 0 早起坐车睡觉颓废报道颓废 反正游记就是咕懒得写了 Day 1 早上四点被xp的闹钟吵醒(???还两次) 幸 ...

  7. .net 基础(一)

    方法 只需要考虑2个 东西 1. 方法的参数  2.方法的返回值 当参数的个数不确定的时候,可以采用可变参数params. params 数组的 个数,不确定.当传入的 参数为空的时候,可变参数的数组 ...

  8. 十七、IntelliJ IDEA 中的 Maven 项目初体验及搭建 Spring MVC 框架

    我们已经将 IntelliJ IDEA 中的 Maven 项目的框架搭建完成.接着上文,在本文中,我们更近一步,利用 Tomcat 运行我们的 Web 项目. 如上图所示,我们进一步扩展了项目的结构, ...

  9. 用Java+xml配置方式实现Spring数据事务(编程式事务)

    一.用Java配置的方式 1.实体类: Role public class Role { private int id; private String roleName; private String ...

  10. 创建oracle数据库表空间并分配用户

    我们在本地的oracle上或者virtualbox的oracle上 创建新的数据库表空间操作:通过system账号来创建并授权/*--创建表空间create tablespace YUJKDATAda ...