[CERC2016]机棚障碍 Hangar Hurdles(kruskal重构树+树上倍增)
题面
$ solution: $
某蒟蒻的心路历程:
这一题第一眼感觉很奇怪
带障碍物的图,最大的集装箱?
首先想到的就是限制我集装箱大小条件的是什么:
如果我要在某一个点上放一个集装箱且使它最大,
那就相当于求这一个点往外扩出去的最大正方形。
然后考虑从一个点到与它相邻的点移动时最大的箱子,
这不就是这两个点的最大正方形中较小的那一个吗?
于是考虑把所有点上的最大正方形都求出来,这个n^2即可
然后我就是要求两个点之间所有路径上的最小的那个最大正方形了
仔细一想,嗯?,这不就是货车运输吗?难道我要切题了?
但毕竟被虐的太多了,还是理智的看一遍吧
哇,边数最多可以达到1000000条,我记得火车运输范围只有10000来着?
我靠,要凉啊,算了直接写暴力得了!(然而,……..)
这一道题,仔细审一下题面我们可以发现:在每一个位置上箱子都有一个初始大小,而它如果要移动到与它中心相邻的某一个点上,他的大小又会受到它要移动到的那个点。所以我们可以先跑一遍BFS来求出没一点的初始最大正方形(这个我们可以反过来,从障碍物出发向四周扩展,每个点只会进队一次,复杂度完全可以接受)。然后我们对任何一对相邻的点连边,边权即为两个中最大正方形较小的那一个。这样我们如果要求两条点可以运输的最大正方形,就是求他们之间所有路径中最小权值最大的那一个,而这不就是货车运输吗????(好吧,其实LCA十分容易被卡,所以建议用树链剖分)
$ code: $
#include<iostream>
#include<cstdio>
#include<iomanip>
#include<algorithm>
#include<cstring>
#include<cstdlib>
#include<ctime>
#include<cmath>
#include<vector>
#include<queue>
#include<map>
#include<set>
#define ll long long
#define db double
#define inf 0x7fffffff
#define rg register int
using namespace std;
struct su{
int x,y,v;
bool operator <(su x){return v>x.v;}
}b[2000005];
struct pi{
int x,y,v;
}t[2000005];
struct ya{
int to,v,next;
}d[2000005];
int n,m,tt,top,ttt,xx,yy,ans;
int a[1000005];
int tou[1000005];
int s[1005][1005];
bool k[1005][1005];
char c[1005];
bool vis[1000005];
int dp[1000005];
int lg[1000005];
int f[1000005][20];
int st[1000005][20];
inline int qr(){
char ch;
while((ch=getchar())<'0'||ch>'9');
int res=ch^48;
while((ch=getchar())>='0'&&ch<='9')
res=res*10+(ch^48);
return res;
}
inline int max(int x,int y){return x<y?y:x;}
inline int min(int x,int y){return x<y?x:y;}
inline int minn(int x,int y,int z){return x<y?(x<z?x:z):(y<z?y:z);}
inline int get(int x){return x==a[x]?x:a[x]=get(a[x]);}
inline void qw(int x){
printf("(%d %d)",(x%n==0?x/n-1:x/n),(x%n==0?n:x%n));
}
inline void yu(){
int i=0,x,y,v;
while(++i<=tt){
x=t[i].x;y=t[i].y;v=t[i].v;
if(!k[x+1][y+1]&&x+1<=n&&y+1<=n) s[x+1][y+1]=v, k[x+1][y+1]=1, t[++tt]=pi{x+1,y+1,v+1};
if(!k[x+1][y]&&x+1<=n&&y<=n) s[x+1][y]=v, k[x+1][y]=1, t[++tt]=pi{x+1,y,v+1};
if(!k[x+1][y-1]&&x+1<=n&&0<y-1) s[x+1][y-1]=v, k[x+1][y-1]=1, t[++tt]=pi{x+1,y-1,v+1};
if(!k[x][y+1]&&0<x&&y+1<=n) s[x][y+1]=v, k[x][y+1]=1, t[++tt]=pi{x,y+1,v+1};
if(!k[x][y-1]&&0<x&&0<y-1) s[x][y-1]=v, k[x][y-1]=1, t[++tt]=pi{x,y-1,v+1};
if(!k[x-1][y+1]&&0<x-1&&y+1<=n) s[x-1][y+1]=v, k[x-1][y+1]=1, t[++tt]=pi{x-1,y+1,v+1};
if(!k[x-1][y]&&0<x-1&&y<=n) s[x-1][y]=v, k[x-1][y]=1, t[++tt]=pi{x-1,y,v+1};
if(!k[x-1][y-1]&&0<x-1&&0<y-1) s[x-1][y-1]=v, k[x-1][y-1]=1, t[++tt]=pi{x-1,y-1,v+1};
}
}
inline bool dfs(int i,int fa,int dp){
for(rg j=tou[i];j;j=d[j].next){
int to=d[j].to;
if(to==fa)continue;
//qw(i);cout<<"=>";qw(to);;
if(to==yy){
ans=min(dp,d[j].v);
return 1;
}
if(dfs(to,i,min(dp,d[j].v)))return 1;
}return 0;
}
inline void lca(int x,int dep){
dp[x]=dep; vis[x]=1;
for(rg i=0;i<19;++i)
f[x][i+1]=f[f[x][i]][i];
for(rg i=0;i<19;++i)
st[x][i+1]=min(st[x][i],st[f[x][i]][i]);
for(rg i=tou[x];i;i=d[i].next){
int to=d[i].to;
if(to==x||vis[to])continue;
f[to][0]=x;
st[to][0]=d[i].v;
lca(to,dep+1);
}
}
inline int find(int x,int y){
if(dp[x]<dp[y])swap(x,y);
ans=100001;
for(rg i=19;i>=0;--i){
if(dp[f[x][i]]>=dp[y]){
ans=min(st[x][i],ans);
x=f[x][i];
}
if(x==y)return ans;
}
for(rg i=19;i>=0;--i){
if(f[x][i]!=f[y][i]){
ans=minn(ans,st[x][i],st[y][i]);
x=f[x][i];y=f[y][i];
}
}
return minn(ans,st[x][0],st[y][0]);
}
int main(){
//freopen("B.in","r",stdin);
//freopen("B.out","w",stdout);
n=qr();
for(rg i=1;i<=n;++i){
scanf("%s",c+1);
for(rg j=1;j<=n;++j)
if(c[j]!='.')k[i][j]=1,t[++tt]=pi{i,j,1};
}
for(rg i=0;i<=n+1;++i){
k[i][0]=1;t[++tt]=pi{i,0,1};
k[i][n+1]=1;t[++tt]=pi{i,n+1,1};
k[0][i]=1;t[++tt]=pi{0,i,1};
k[n+1][i]=1;t[++tt]=pi{n+1,i,1};
}yu();
for(rg i=1;i<=n;++i)
for(rg j=1;j<n;++j)
b[++top]=su{i*n+j-n,i*n+j+1-n,min(s[i][j],s[i][j+1])};
for(rg i=1;i<n;++i)
for(rg j=1;j<=n;++j)
b[++top]=su{i*n+j-n,i*n+j,min(s[i][j],s[i+1][j])};
sort(b+1,b+top+1);
for(rg i=1;i<=n*n;++i)a[i]=i;
for(rg i=1;i<=top;++i){
if(get(b[i].x)!=get(b[i].y)){
a[get(b[i].x)]=get(b[i].y);
d[++ttt]=ya{b[i].y,b[i].v,tou[b[i].x]};tou[b[i].x]=ttt;
d[++ttt]=ya{b[i].x,b[i].v,tou[b[i].y]};tou[b[i].y]=ttt;
}
}
m=qr(); lca(1,1);
for(rg i=1;i<=m;++i){
xx=qr()*n+qr()-n; yy=qr()*n+qr()-n;
int ans=find(xx,yy);
printf("%d\n",(!ans?0:ans*2-1));
}
return 0;
}
[CERC2016]机棚障碍 Hangar Hurdles(kruskal重构树+树上倍增)的更多相关文章
- LG3684 [CERC2016]机棚障碍 Hangar Hurdles
题意 题目描述 你正在评估一些关于一个巨型飞机仓库的建设计划.飞机仓库的地面可以表示为n行n列的网格图,其中每个格子要么是空的,要么有障碍物.行从上到下依次被编号为1到n,列从左到右依次被编号为1到n ...
- BZOJ.4793.[CERC2016]Hangar Hurdles(Kruskal重构树 BFS)
题目链接 \(Description\) 有一个\(n\times n\)的正方形网格,上面有若干障碍点.\(q\)次询问,每次询问把一个正方形箱子从\((x1,y1)\)推到\((x2,y2)\) ...
- Gym - 101173H Hangar Hurdles (kruskal重构树/最小生成树+LCA)
题目大意:给出一个n*n的矩阵,有一些点是障碍,给出Q组询问,每组询问求两点间能通过的最大正方形宽度. 首先需要求出以每个点(i,j)为中心的最大正方形宽度mxl[i][j],可以用二维前缀和+二分或 ...
- 【BZOJ 3551】[ONTAK2010] Peaks加强版 Kruskal重构树+树上倍增+主席树
这题真刺激...... I.关于Kruskal重构树,我只能开门了,不过补充一下那玩意还是一棵满二叉树.(看一下内容之前请先进门坐一坐) II.原来只是用树上倍增求Lca,但其实树上倍增是一种方法,L ...
- 洛谷 P3684 机棚障碍Hangar Hurdles [CERC2016] 图论
正解: 解题报告: 传送门! 首先不难想到这题主要有两个问题需要解决,一个是预处理出各个点的箱子半径最大值,一个是求ans 然后分别港下QwQ 首先关于预处理要说下昂 预处理有三种方法,分别港下 第一 ...
- UVA1265 Tour Belt Kruskal重构树、倍增、树上差分
题目传送门 题意:定义$Tour \, Belt$为某张图上的一个满足以下条件的点集:①点集中至少有$2$个点②任意两点互相连通③图上两个端点都在这个点集中的边的权值的最小值严格大于图上只有一个端点在 ...
- Kruskal 重构树小记
其实也不是多难的知识点吧--学了一个中午+半个下午就把它学会了(做过那道 jxd 作业 CF571D 的应该比较好理解) Kruskal 重构树大概就是在正常 Kruskal 的时候,对于两个需要连边 ...
- 【BZOJ-3545&3551】Peaks&加强版 Kruskal重构树 + 主席树 + DFS序 + 倍增
3545: [ONTAK2010]Peaks Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1202 Solved: 321[Submit][Sta ...
- 洛谷P4768 [NOI2018]归程(Kruskal重构树)
题意 直接看题目吧,不好描述 Sol 考虑暴力做法 首先预处理出从$1$到每个节点的最短路, 对于每次询问,暴力的从这个点BFS,从能走到的点里面取$min$ 考虑如何优化,这里要用到Kruskal重 ...
随机推荐
- ansible系列6-用户管理
第一种:新增用户 ansible host -m user -a "name=zhang shell=/bin/bash groups=admin,root append=yes home= ...
- BZOJ3963 WF2011MachineWorks(动态规划+斜率优化+cdq分治)
按卖出时间排序后,设f[i]为买下第i台机器后的当前最大收益,则显然有f[i]=max{f[j]+gj*(di-dj-1)+rj-pi},且若此值<0,应设为-inf以表示无法购买第i台机器. ...
- Python3网络爬虫(3):使用User Agent和代理IP隐藏身份
Python版本: python3 IDE: pycharm2017.3.3 一.为何要设置User Agent 有一些网站不喜欢被爬虫访问,所以会检测对象,如果是爬虫程序,他就会不让你访问,通过设置 ...
- 【刷题】BZOJ 2407 探险
Description 探险家小T好高兴!X国要举办一次溶洞探险比赛,获奖者将得到丰厚奖品哦!小T虽然对奖品不感兴趣,但是这个大振名声的机会当然不能错过! 比赛即将开始,工作人员说明了这次比赛的规则: ...
- 【转】Linux 移动或重命名文件/目录-mv 的10个实用例子
熟悉了 复制命令,下一个相关的命令就是 mv 命令.当你想要将文件从一个位置移动到另一个地方并且不想复制它,那么mv 命令是完成这个任务的首选. 初识 mv 命令 mv 命令是一个与cp类似的命令,但 ...
- 利用spring boot+vue做的一个博客项目
技术栈: 后端 Springboot druid Spring security 数据库 MySQL 前端 vue elementUI 项目演示: GitHub地址: 后端:https://githu ...
- bug6 项目检出JRE问题(Unbound classpath container: 'JRE System Library [JavaSE-1.7]' in project 'idweb')
项目从SVN检出到工作空间后报了很多错误,其中很明显就是一些jar的问题,没有相关的jar或版本问题,看到最后的错误Unbound classpath Container: 'JRE System L ...
- Faster RCNN代码理解(Python)
转自http://www.infocool.net/kb/Python/201611/209696.html#原文地址 第一步,准备 从train_faster_rcnn_alt_opt.py入: 初 ...
- 盘点 php 里面那些冷门又实用的小技巧
1.实用某个字段索引二维数组 取出一个数组的一个字段的值的数组,我们可以使用 array_column, 这个方法还有另外一个用法,如 array_column($array, null, 'key' ...
- nginx 中配置多个location并解决js/css/jpg/等的加载问题
2017-11-09 22:07 277人阅读 评论(0) 收藏 举报 分类: linux(1) 版权声明:如有版权问题,请私信我. ECS:阿里云 系统:ubuntu 16.04 我的配置文件位 ...