题意:

当你是人形的时候你只能走 \([L,N-1]\) 的编号的点(即大于等于L的点)

当你是狼形的时候你只能走 \([1,R]\) 的编号的点(即小于等于R的点)

然后问题转化成人形和狼形能到的点有没有交集。

solution:

发现可以建 kruskal重构树,就可以通过在树上倍增来求出来一个子树,这个子树内是你可以到的点。然后问题转化成了两个子树区间的交,这个问题可以用主席树解决。

定义 \(rev1_i\) 是在树上的第 \(i\) 个位置对应的数,那你定义 \(r_{rev1_i}=i\),那么 \(r_i\) 就是这个元素在树上出现的位置,\(rev2_i\)同理,即这个数在第二棵树上对应的数,然后主席树求解。

#include <bits/stdc++.h>
using namespace std; int read() {
int x = 0;
char c = getchar();
while (c < 48) c = getchar();
while (c > 47) x = x * 10 + (c - 48), c = getchar();
return x;
} int n, m, q;
const int maxn = 4e5 + 54; struct edge {
int u, v;
};
vector<edge> e; int fa[maxn];
int find(int x) { return x == fa[x] ? x : fa[x] = find(fa[x]); } vector<int> g[maxn]; struct Tree {
int L[maxn], R[maxn], idx;
int rev[maxn], val[maxn], f[maxn][22];
void dfs(int u) {
if (u <= n) {
rev[L[u] = ++idx] = u;
val[u] = u;
} else
L[u] = idx + 1;
for (int i = 1; i <= 20; i++) f[u][i] = f[f[u][i - 1]][i - 1];
for (int v : g[u]) f[v][0] = u, dfs(v);
R[u] = idx;
}
int getu(int s, int l) {
for (int i = 20; ~i; i--)
if (f[s][i] && val[f[s][i]] >= l) s = f[s][i];
return s;
}
int getv(int t, int r) {
for (int i = 20; ~i; i--)
if (f[t][i] && val[f[t][i]] <= r) t = f[t][i];
return t;
}
} t1, t2; int tot;
int rt[maxn], ls[maxn << 5], rs[maxn << 5], val[maxn << 5], cnt = 0; void upd(int& p, int pre, int l, int r, int x) {
ls[p = ++cnt] = ls[pre], rs[p] = rs[pre], val[p] = val[pre] + 1;
if (l == r) return;
int mid = l + r >> 1;
if (x <= mid)
upd(ls[p], ls[pre], l, mid, x);
else
upd(rs[p], rs[pre], mid + 1, r, x);
} int qry(int a, int b, int ql, int qr, int l, int r) {
if (ql <= l && r <= qr) return val[b] - val[a];
int mid = l + r >> 1, ans = 0;
if (ql <= mid) ans = qry(ls[a], ls[b], ql, qr, l, mid);
if (qr > mid) ans += qry(rs[a], rs[b], ql, qr, mid + 1, r);
return ans;
} int main() {
n = read(), m = read(), q = read();
for (int i = 0; i < m; i++) {
int u = read(), v = read();
++u, ++v;
if (u > v) u ^= v ^= u ^= v;
e.push_back({ u, v });
}
for (int i = 1; i <= n * 2; i++) fa[i] = i, g[i].clear();
sort(e.begin(), e.end(), [](edge a, edge b) { return a.u > b.u; });
tot = n;
for (auto x : e) {
int u = find(x.u), v = find(x.v);
if (u == v) continue;
fa[u] = fa[v] = ++tot;
t1.val[tot] = x.u;
g[tot].push_back(u), g[tot].push_back(v);
}
t1.dfs(tot); for (int i = 1; i <= n * 2; i++) fa[i] = i, g[i].clear();
sort(e.begin(), e.end(), [](edge a, edge b) { return a.v < b.v; });
tot = n;
for (auto x : e) {
int u = find(x.u), v = find(x.v);
if (u == v) continue;
fa[u] = fa[v] = ++tot;
t2.val[tot] = x.v;
g[tot].push_back(u), g[tot].push_back(v);
}
t2.dfs(tot);
vector<int> r(n + 1);
for (int i = 1; i <= n; i++) r[t1.rev[i]] = i;
for (int i = 1; i <= n; i++) upd(rt[i], rt[i - 1], 1, n, r[t2.rev[i]]);
for (int i = 0; i < q; i++) {
int s = read(), t = read(), l = read(), r = read();
++s, ++t, ++l, ++r;
int u = t1.getu(s, l), v = t2.getv(t, r);
if (qry(rt[t2.L[v] - 1], rt[t2.R[v]], t1.L[u], t1.R[u], 1, n))
puts("1");
else
puts("0");
}
return 0;
}

[IOI2018] werewolf 狼人 [kruskal重构树+主席树]的更多相关文章

  1. [IOI2018] werewolf 狼人 kruskal重构树,主席树

    [IOI2018] werewolf 狼人 LG传送门 kruskal重构树好题. 日常安利博客文章 这题需要搞两棵重构树出来,这两棵重构树和我们平时见过的重构树有点不同(据说叫做点权重构树?),根据 ...

  2. LOJ.2865.[IOI2018]狼人(Kruskal重构树 主席树)

    LOJ 洛谷 这题不就是Peaks(加强版)或者归程么..这算是\(IOI2018\)撞上\(NOI2018\)的题了? \(Kruskal\)重构树(具体是所有点按从小到大/从大到小的顺序,依次加入 ...

  3. luoguP4197:Peaks(Kruskal重构树+主席树)或者(点分树+离线)

    题意:有N座山,M条道路.山有山高,路有困难值(即点权和边权).现在Q次询问,每次给出(v,p),让求从v出发,只能结果边权<=p的边,问能够到达的山中,第K高的高度(从大到小排序). 思路:显 ...

  4. UOJ#407. 【IOI2018】狼人 Kruskal,kruskal重构树,主席树

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ407.html 题解 套路啊. 先按照两个节点顺序各搞一个kruskal重构树,然后问题转化成两棵krus ...

  5. 【BZOJ-3545&3551】Peaks&加强版 Kruskal重构树 + 主席树 + DFS序 + 倍增

    3545: [ONTAK2010]Peaks Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1202  Solved: 321[Submit][Sta ...

  6. BZOJ3545&3551[ONTAK2010]Peaks——kruskal重构树+主席树+dfs序+树上倍增

    题目描述 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困难值,这个值越大表示越难走,现在有Q组询问,每组询问询问从点v开始只 ...

  7. luogu4197 Peaks (kruskal重构树+主席树)

    按照边权排序建出kruskal重构树,每次就变成了先找一个权值<=x的最远的祖先,然后看这个子树的第k小.离散化一下,在dfs序上做主席树即可 而且只需要建叶节点的主席树 注意输出的是第k小点的 ...

  8. 洛谷P4197 Peaks(Kruskal重构树 主席树)

    题意 题目链接 往后中文题就不翻译了qwq Sol 又是码农题..出题人这是强行把Kruskal重构树和主席树拼一块了啊.. 首先由于给出的限制条件是<=x,因此我们在最小生成树上走一定是最优的 ...

  9. [BZOJ3551][ONTAK2010]Peaks(加强版)(Kruskal重构树,主席树)

    3551: [ONTAK2010]Peaks加强版 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 2438  Solved: 763[Submit][ ...

随机推荐

  1. Intellij IDEA使用(一)项目模板类型

    概述 IDEA默认打开后,会出现欢迎页面,可以选择打开Project的方式. 对于新建的Project,一般会选择[Create New Project],对于界面中,众多的Project类型,有时候 ...

  2. 老哈佛H3的空调控制器的维修记录

    这部哈弗车是2005年出厂的,应该差不多是第一批了吧,小毛病很多了.夏天到了,空调也不能启动,灯不亮,按键没反应令我很是着急.于是开始研究. 这款空调的控制器在cd机中控下面,需要拆中控取出.是一个黑 ...

  3. springboot整合elasticJob实战(纯代码开发三种任务类型用法)以及分片系统,事件追踪详解

    一 springboot整合 介绍就不多说了,只有这个框架是当当网开源的,支持分布式调度,分布式系统中非常合适(两个服务同时跑不会重复,并且可灵活配置分开分批处理数据,贼方便)! 这里主要还是用到zo ...

  4. Docker深入浅出系列 | 单节点多容器网络通信

    目录 教程目标 准备工作 带着问题开车 同一主机两个容器如何相互通信? 怎么从服务器外访问容器 Docker的三种网络模式是什么 Docker网络通信原理 计算机网络模型回顾 Linux中的网卡 查看 ...

  5. 【WPF学习】第三十五章 资源字典

    如果希望在多个项目之间共享资源,可创建资源字典.资源字典只是XAML文档,除了存储希望使用的资源外,不做其他任何事情. 一.创建资源字典 下面是一个资源字典示例,它包含一个资源: <Resour ...

  6. ros机器人之小乌龟仿真-路径记录

    ------------恢复内容开始------------ 通过自己不断地摸索,对ros系统有了一定的了解,首先装系统,这一过程中也遇到了很多问题,但通过不断地尝试,经过一天一夜的倒腾,总算是把系统 ...

  7. 如何优雅地使用containerd?这里有一份必读的技巧攻略

    前 言 Docker是我们常用的容器runtime,友好的CLI,丰富的社区资料,外加研发运维人员多年的经验积累,使用Docker几乎是没有任何门槛的事.而k3s为了降低资源消耗,将默认的runtim ...

  8. OpenGL学习网址2

    http://www.opengpu.org/forum.php?mod=viewthread&tid=7525

  9. Codeforces_731_A

    http://codeforces.com/problemset/problem/731/A 每次操作总共4种情况,判断一下就好了. #include<iostream> #include ...

  10. centos下利用phantomjs来完成网站页面快照截图

    最近研究了下phantomjs,感觉还是非常不错的. 首先到官网下载一个源码包 http://phantomjs.org/download.html 点击源码包下载如图: 然后在linux下将必要的一 ...