题目传送门

题意:

  给出m条无向边,每条边都有一个$[l,r]$,意思是体积在这个范围内的人才能通过这条边,询问有多少种体积的可能性,能使人从1到n

思路:由于是无向边,1和n的连通性可以用并查集维护。

  考虑最暴力的做法,枚举每一种体积,将当前体积能通过的边用并查集维护一下,判断1到n的连通性即可。

  考虑优化,首先区间必定可以离散化,并且我们可以将离散化后的区间按照线段树的方式划分一下,然后将每一个子区间的边用并查集连一下,判断连通性,然后离开这个子区间时,将并查集的连边操作退回就可以了。

  由于每条边最多被划分成log个区间,也就是会被并查集加入log次,拆开log次,所以时间复杂度是(n*logn*logn),其中一个log是并查集带来的,常数很小。

  可撤销的并查集用按秩合并来完成,并且只能回退连续一段时间的操作,离散化后点有2*n个,线段树点不要开太少。

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define dep(i,b,a) for(int i=b;i>=a;i--)
#define pb(x) push_back(x)
#define pii pair<int,int >
using namespace std;
const int maxn=;
vector<int >tr[maxn<<];
int n,m;
struct edge{
int u,v,l,r;
}a[maxn];
int b[maxn<<],cnt,ans;
int fa[maxn],siz[maxn];
int find(int x){
return x==fa[x]?x:find(fa[x]);
}
void update(int o,int l,int r,int ql,int qr,int i){
if(ql<=l&&r<=qr){
tr[o].pb(i);
return;
}
int mid=(l+r)>>;
if(ql<=mid)update(o<<,l,mid,ql,qr,i);
if(mid<qr)update(o<<|,mid+,r,ql,qr,i);
}
void dfs(int o,int l,int r){
vector<pii >ve;
int si=tr[o].size();
rep(i,,si-){
int id=tr[o][i];
int u=a[id].u,v=a[id].v;
int fu=find(u),fv=find(v);
if(fu==fv)continue;
if(siz[fu]>siz[fv])swap(fu,fv);
fa[fu]=fv;
int d=;
if(siz[fu]==siz[fv])d++;
siz[fv]+=d;
ve.push_back({fu,d});
}
int mid=(l+r)>>;
if(find()==find(n)){
ans+=b[r+]-b[l];
}else if(l<r){
dfs(o<<,l,mid);
dfs(o<<|,mid+,r);
}
si=ve.size();
dep(i,si-,){
siz[fa[ve[i].first]]-=ve[i].second;
fa[ve[i].first]=ve[i].first;
}
}
int main(){
cin>>n>>m;
rep(i,,n){
fa[i]=i;
siz[i]=;
}
for(int i=;i<=m;i++){
scanf("%d%d%d%d",&a[i].u,&a[i].v,&a[i].l,&a[i].r);
b[++cnt]=a[i].l;
b[++cnt]=a[i].r+;
}
sort(b+,b++cnt);
cnt=unique(b+,b++cnt)-b-;
for(int i=;i<=m;i++){
a[i].l=lower_bound(b+,b++cnt,a[i].l)-b;
a[i].r=lower_bound(b+,b++cnt,a[i].r+)-b;
update(,,cnt,a[i].l,a[i].r-,i);
}
dfs(,,cnt);
printf("%d\n",ans);
}

2019牛客暑期多校训练营(第八场) E 线段树+可撤销并查集的更多相关文章

  1. 2019牛客暑期多校训练营(第九场)All men are brothers——并查集&&组合数

    题意 最初有 $n$ 个人且互不认识,接下来 $m$ 行,每行有 $x,y$,表示 $x$ 和 $y$ 交朋友,朋友关系满足自反性和传递性,每次输出当前选取4个人且互不认识的方案数. 分析 并查集维护 ...

  2. 2019牛客暑期多校训练营(第九场)A:Power of Fibonacci(斐波拉契幂次和)

    题意:求Σfi^m%p. zoj上p是1e9+7,牛客是1e9:  对于这两个,分别有不同的做法. 前者利用公式,公式里面有sqrt(5),我们只需要二次剩余求即可.     后者mod=1e9,5才 ...

  3. 2019牛客暑期多校训练营(第一场)A题【单调栈】(补题)

    链接:https://ac.nowcoder.com/acm/contest/881/A来源:牛客网 题目描述 Two arrays u and v each with m distinct elem ...

  4. 2019牛客暑期多校训练营(第一场) B Integration (数学)

    链接:https://ac.nowcoder.com/acm/contest/881/B 来源:牛客网 Integration 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 5242 ...

  5. 2019牛客暑期多校训练营(第一场) A Equivalent Prefixes ( st 表 + 二分+分治)

    链接:https://ac.nowcoder.com/acm/contest/881/A 来源:牛客网 Equivalent Prefixes 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/ ...

  6. 2019牛客暑期多校训练营(第二场)F.Partition problem

    链接:https://ac.nowcoder.com/acm/contest/882/F来源:牛客网 Given 2N people, you need to assign each of them ...

  7. 2019牛客暑期多校训练营(第一场)A Equivalent Prefixes(单调栈/二分+分治)

    链接:https://ac.nowcoder.com/acm/contest/881/A来源:牛客网 Two arrays u and v each with m distinct elements ...

  8. [状态压缩,折半搜索] 2019牛客暑期多校训练营(第九场)Knapsack Cryptosystem

    链接:https://ac.nowcoder.com/acm/contest/889/D来源:牛客网 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 262144K,其他语言52428 ...

  9. 2019牛客暑期多校训练营(第二场)J-Subarray(思维)

    >传送门< 前言 这题我前前后后看了三遍,每次都是把网上相关的博客和通过代码认真看了再思考,然并卵,最后终于第三遍也就是现在终于看懂了,其实懂了之后发现其实没有那么难,但是的的确确需要思维 ...

  10. 2019牛客暑期多校训练营(第一场)-A (单调栈)

    题目链接:https://ac.nowcoder.com/acm/contest/881/A 题意:给定两个长度均为n的数组a和b,求最大的p使得(a1,ap)和(b1,bp)等价,等价的定义为其任意 ...

随机推荐

  1. 你真的懂return吗?

    递归算法中什么时候用return啥时候不用呢? 使用2个例子来说明:快速排序和二分查找 # 二分查找def binarySearch (arr, l, r, x): # 基本判断 if r >= ...

  2. C语言static和局部变量

    #include <stdio.h> void test(); int main() { /************************************************ ...

  3. react-router踩坑

    1.当用BrowserRouter时,<Link/>组件必须放在BrowserRouter里,如果是存在于某个组件里的Link,则该组件也必须放在Router里,不然会出现url改变页面为 ...

  4. uname - 显示输出系统信息

    总览 uname [OPTION]... 描述 显示相应的系统信息. 没有指定选项时,同 -s. -a, --all 显示所有的信息 -m, --machine 显示机器(硬件)类型 -n, --no ...

  5. CentOS7.6编译安装openssl-1.1.1c

    卸载旧版本OpenSSL # which openssl/usr/bin/openssl# mv openssl openssl.oldrm -rf /etc/ssl #删除配置文件 CentOS7. ...

  6. [LOJ#6468.] 魔法

    官方题解 看了题解才会做.. 首先考虑如果所有询问的点都是[1,n]的做法,如果询问是[l,r]只需要把多余的去掉就好了 然后要把问题转化为一个点对其他附近的点的贡献 记$pre[i]$为第i个位置的 ...

  7. [转载]python异常如何全面捕获

    写在前面:最近写python程序,进场遇到异常的问题,因此需要捕获异常.查阅了下资料,整理如下: 常见的异常处理的方法: 假设有下面的一段程序: try:     语句1     语句2     . ...

  8. scip 练习2.20

    (define (same-parity x . z) (define (q? y) (= (remainder y ) )) (define (o? y) (= (remainder y ) )) ...

  9. nodejs route的简单使用

    demo var express=require('express'); var app=express(); var routeUser=express.Router(); var routeTea ...

  10. leetcood学习笔记-172-阶乘后的0

    题目描述: 方法:不断除以 5, 是因为每间隔 5 个数有一个数可以被 5 整除, 然后在这些可被 5 整除的数中, 每间隔 5 个数又有一个可以被 25 整除, 故要再除一次, ... 直到结果为 ...