UOJ_407_【IOI2018】狼人
分析:
- 分别建立最小/最大kruskal重构树。
- 每次询问给出的两个点能走到的部分分别对应两棵树\(dfs\)序的一段区间。
- 转化成判断矩形中是否有点。
代码:
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <algorithm>
#include <cmath>
#include "werewolf.h"
using namespace std;
#define M 400050
#define vi vector<int>
int n,m,q_times,fa[M],ln;
int dfn[2][M],enp[2][M],Ls[2][M],Rs[2][M],dfc,w[2][M];
int f[2][20][M],idf[2][M],root[M],ls[M*22],rs[M*22],siz[M*22],cnt;
char buf[100000],*p1,*p2;
#define nc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++)
int rd() {
int x=0; char s=nc();
while(s<'0') s=nc();
while(s>='0') x=(((x<<2)+x)<<1)+s-'0',s=nc();
return x;
}
struct A {
int a,b,c;
bool operator < (const A &u) const {
return c>u.c;
}
}e[M];
int find(int x) {return fa[x]==x?x:fa[x]=find(fa[x]);}
void df1(int x,int o) {
dfn[o][x]=++dfc;
idf[o][dfc]=x;
if(Ls[o][x]) df1(Ls[o][x],o);
if(Rs[o][x]) df1(Rs[o][x],o);
enp[o][x]=dfc;
}
void update(int l,int r,int x,int &p,int q) {
p=++cnt; siz[p]=siz[q]+1; ls[p]=ls[q]; rs[p]=rs[q];
if(l==r) return ;
int mid=(l+r)>>1;
if(x<=mid) update(l,mid,x,ls[p],ls[q]);
else update(mid+1,r,x,rs[p],rs[q]);
}
int query(int l,int r,int x,int y,int p,int q) {
if(x<=l&&y>=r) return siz[p]-siz[q];
int mid=(l+r)>>1,re=0;
if(x<=mid) re+=query(l,mid,x,y,ls[p],ls[q]);
if(y>mid) re+=query(mid+1,r,x,y,rs[p],rs[q]);
return re;
}
vi check_validity(int n,vi EX,vi EY,vi QX,vi QY,vi QL,vi QR) {
ln=n*2-1;
m=EX.size();
int i,tot=n;
for(i=1;i<=m;i++) {
e[i].a=EX[i-1]; e[i].b=EY[i-1];
e[i].a++; e[i].b++;
e[i].c=min(e[i].a,e[i].b);
}
sort(e+1,e+m+1);
for(i=1;i<=ln;i++) fa[i]=i;
for(i=1;i<=m;i++) {
int dx=find(e[i].a),dy=find(e[i].b);
if(dx!=dy) {
fa[dx]=fa[dy]=++tot; Ls[0][tot]=dx; Rs[0][tot]=dy; w[0][tot]=e[i].c; f[0][0][dx]=f[0][0][dy]=tot;
}
}
df1(tot,0); dfc=0;
for(i=1;i<=ln;i++) fa[i]=i;
tot=n;
for(i=1;i<=m;i++) e[i].c=max(e[i].a,e[i].b);
sort(e+1,e+m+1);
for(i=m;i;i--) {
int dx=find(e[i].a),dy=find(e[i].b);
if(dx!=dy) {
fa[dx]=fa[dy]=++tot; Ls[1][tot]=dx; Rs[1][tot]=dy; w[1][tot]=e[i].c; f[1][0][dx]=f[1][0][dy]=tot;
}
}
df1(tot,1);
int j,k;
for(i=0;i<2;i++) {
for(j=1;(1<<j)<=ln;j++) {
for(k=1;k<=ln;k++) {
f[i][j][k]=f[i][j-1][f[i][j-1][k]];
}
}
}
for(i=1;i<=dfc;i++) {
root[i]=root[i-1];
int x=idf[1][i];
if(x<=n) {
update(1,dfc,dfn[0][x],root[i],root[i]);
}
}
vi ans;
int x,y,l,r;
q_times=QX.size();
for(j=0;j<q_times;j++) {
x=QX[j]; y=QY[j]; l=QL[j]; r=QR[j];
x++;y++;l++;r++;
for(i=19;i>=0;i--) {
if(f[0][i][x]&&w[0][f[0][i][x]]>=l) x=f[0][i][x];
if(f[1][i][y]&&w[1][f[1][i][y]]<=r) y=f[1][i][y];
}
int t=query(1,dfc,dfn[0][x],enp[0][x],root[enp[1][y]],root[dfn[1][y]-1]);
ans.push_back(t>0);
}
return ans;
}
UOJ_407_【IOI2018】狼人的更多相关文章
- [IOI2018]狼人
[IOI2018]狼人 luogu UOJ 对人形和狼形分别建克鲁斯卡尔重构树 每次询问就是对于两棵树dfs序的一个二维数点,主席树维护 #include<bits/stdc++.h> u ...
- LOJ.2865.[IOI2018]狼人(Kruskal重构树 主席树)
LOJ 洛谷 这题不就是Peaks(加强版)或者归程么..这算是\(IOI2018\)撞上\(NOI2018\)的题了? \(Kruskal\)重构树(具体是所有点按从小到大/从大到小的顺序,依次加入 ...
- [IOI2018]狼人——kruskal重构树+可持久化线段树
题目链接: IOI2018werewolf 题目大意:给出一张$n$个点$m$条边的无向图,点和边可重复经过,一个狼人初始为人形,有$q$次询问,每次询问要求人形态只能处于编号不小于$L$的点,狼形态 ...
- Luogu4899 IOI2018狼人(kruskal重构树+主席树)
可以发现询问的即是“由起点开始‘只经过编号大于等于l的点’所形成的连通块”与“由终点开始‘只经过编号小于等于r的点’所形成的连通块”是否有交集.于是建出重构树,就可以知道每个询问的连通情况了.现在要知 ...
- ghj1222被坑记录[不持续更新]
考试注意事项:link1 link2 (密码:wangle) 调不出来bug,可以先透彻一会儿或者是上个厕所或者坐一会别的题(间隔至少20min),然后通读代码 -1. 考试先读题,读题之后搞出一个做 ...
- [note]克鲁斯卡尔重构树
克鲁斯卡尔重构树 又叫并查集重构树 大概在NOI2018之前还是黑科技 现在?烂大街了 主要是针对图上的对边有限制的一类问题 比如每次询问一个点u不能经过边权大于w的边能走到的第k大点权是多少 也就是 ...
- NOIWC2019游记
更新完了? ghj1222这个智障因为NOIP考的太菜没有去THUWC和PKUWC,但是NOIWC还是苟进去了 由于已经结束了,好多事实忘了,所以可能不完整 2019/1/23 Wednesday 明 ...
- [IOI2018] werewolf 狼人
[IOI2018] werewolf 狼人 IOI2018题解 (其实原题强制在线,要用主席树) 代码: 注意: 1.下标从0~n-1 2.kruskal重构树开始有n个节点,tot从n开始,++to ...
- [IOI2018] werewolf 狼人 kruskal重构树,主席树
[IOI2018] werewolf 狼人 LG传送门 kruskal重构树好题. 日常安利博客文章 这题需要搞两棵重构树出来,这两棵重构树和我们平时见过的重构树有点不同(据说叫做点权重构树?),根据 ...
随机推荐
- Spring学习九----------Bean的配置之Bean的定义及作用域的注解实现
© 版权声明:本文为博主原创文章,转载请注明出处 Spring Bean常用注解 @Component:通常注解,可用于任何Bean @Repository:通常用于注解DAO层,即持久层 @Serv ...
- LOL英雄联盟代打外挂程序-java实现
相信非常多程序员都玩游戏,比方LOL :有时候想打人机对战(玩家对战小心别人举报你! ),纯属为了拿经验和金币,而本身不想玩,但假设玩家不操作.那么非常快就会被系统觉得是挂机,从而得不到经验和金币.所 ...
- C++成员不通过对象调用的直接调用写法
C++成员不通过对象调用(.或->方式)的另类(C式)调用写法 #include <iostream> using namespace std; /* 我们知道,成员函数和普通函数最 ...
- mongodb的启动参数--quiet
”mongo群友在群里问了个问题,问的是--quiet启动参数如何用的? 如何理解安静的输出?“ 看到这个问题,之前看过--quiet这个参数,没有认真研究过,也没在生产中使用过. 在mongodb启 ...
- vagrant 介绍,安装与使用
可以帮你统一团队成员的开发环境.如果你或者你的伙伴创建了一个Vagrantfile,那么你只需要执行vagrant up就行了,所有的软件都会安装并且配置好.团队成员可以通过相同的Vagrantfil ...
- pycharm 5 注册码
BIG3CLIK6F-eyJsaWNlbnNlSWQiOiJCSUczQ0xJSzZGIiwibGljZW5zZWVOYW1lIjoibGFuIHl1IiwiYXNzaWduZWVOYW1lIjoiI ...
- 【JavaSE】Java问题总结
使用BufferedInputStream时OutOfMemoryError异常 eclipse Luna安装subversion(SVN) 使用BufferedInputStream时OutOfMe ...
- JS中try.. catch..的用法
try 测试代码块的错误. catch 语句处理错误. throw 创建并跑出错误. try { //在这里运行代码 抛出错误 } catch(err) { //在这里处理错误 } 下面是一个实例: ...
- 【BZOJ3489】A simple rmq problem kd-tree
[BZOJ3489]A simple rmq problem Description 因为是OJ上的题,就简单点好了.给出一个长度为n的序列,给出M个询问:在[l,r]之间找到一个在这个区间里只出现过 ...
- 我的Java开发学习之旅------>Java NIO 报java.nio.charset.MalformedInputException: Input length = 1异常
今天在使用Java NIO的Channel和Buffer进行文件操作时候,报了java.nio.charset.MalformedInputException: Input length = 1异常, ...