不想翻译了,直接放luogu翻译


说了没有偶环,也就是说全是奇环,再结合二分图性质,那么暴力的话,固定左端点,增大序号,加点直到产生环就不合法了。也就是说,任何一个环,只要他上面的数全都被加了,就不合法了,那么,环上的数若最大$\text{m2}$,最小$\text{m1}$,那么如果当前枚举的子区间$x\le m1且y\ge m2$,那就不合法了。这样,我们可以转化一下问题,给一堆区间,然后每次问$[L,R]$有多少子区间不合法(即存在一个给定的区间被这个子区间覆盖了),然后总方案数减去不合法数。但是还存在两个问题。

一,上述的环有多少个?怎么找?发现所有环都是奇环,任何两个奇环不能相交或者公用边,因为这些情况都会附带再产生一个偶环(结论源自于手动模拟),与题意矛盾。所以,每条边最多属于一个简单的奇环。那么,这就是一个仙人掌。。所以,可以证出,环是相互独立的,并且不会超过$M/3$个。这样,就可以接受了,这些环通过点连接,可以看成是一堆点双连接,我们直接tarjan找点双,每个点双对应到区间上即可。

二,区间询问怎么做?`````假设左端点$x$,那么右端点向右开始挪,当不合法(也就是区间被覆盖了)的时候后面都不合法了。再看以$x-1$为左端点,首先不合法的右端点只会比$x$的小。。所以,我们维护这样一个$rb_i$表示最早找到的一个右端点使得这个区间$[i,rb_i]$覆盖了某给定区间,那么他是单调不减的(就是上面这个说明)。那么,为了计算所有不合法子区间,只要计算$\sum\limits_{i\in [l,r],rb_i\le r}(r-rb_i+1)=(x-l+1)(r+1)-\sum\limits_{i=l}^{x}rb_i$。这里的$x$是最靠右的$i$,且$i\in [l,r],rb_i\le r$。为了找到$x$,利用上述$rb$单调性质,直接二分查找即可。sum的部分可以直接前缀和预处理。然后即可快速计算答案。

有关仙人掌的坑。。有空再补。。虽然这题似乎和仙人掌没啥太大关系

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
#define mst(x) memset(x,0,sizeof x)
#define dbg(x) cerr << #x << " = " << x <<endl
#define dbg2(x,y) cerr<< #x <<" = "<< x <<" "<< #y <<" = "<< y <<endl
using namespace std;
typedef long long ll;
typedef double db;
typedef pair<int,int> pii;
template<typename T>inline T _min(T A,T B){return A<B?A:B;}
template<typename T>inline T _max(T A,T B){return A>B?A:B;}
template<typename T>inline char MIN(T&A,T B){return A>B?(A=B,):;}
template<typename T>inline char MAX(T&A,T B){return A<B?(A=B,):;}
template<typename T>inline void _swap(T&A,T&B){A^=B^=A^=B;}
template<typename T>inline T read(T&x){
x=;int f=;char c;while(!isdigit(c=getchar()))if(c=='-')f=;
while(isdigit(c))x=x*+(c&),c=getchar();return f?x=-x:x;
}
const int N=3e5+;
struct thxorz{
int to[N<<],nxt[N<<],head[N],tot;
inline void add(int x,int y){
to[++tot]=y,nxt[tot]=head[x],head[x]=tot;
to[++tot]=x,nxt[tot]=head[y],head[y]=tot;
}
}G;
ll sum[N];
int n,m,q,L,R,l,r;
int rb[N];
#define y G.to[j]
int dfn[N],low[N],tim,stk[N],Top,rt;
void tarjan(int x){//dbg(x);
dfn[x]=low[x]=++tim;
if(rt==x&&!G.head[x])return;
stk[++Top]=x;
for(register int j=G.head[x];j;j=G.nxt[j]){
if(!dfn[y]){
tarjan(y),MIN(low[x],low[y]);
if(low[y]>=dfn[x]){
int tmp,mi=n+,ma=,cnt=;
do MIN(mi,tmp=stk[Top--]),MAX(ma,tmp),++cnt;while(tmp^y);
MIN(mi,x),MAX(ma,x),++cnt;//dbg2(mi,ma);
if(cnt>)MIN(rb[mi],ma);
}
}
else MIN(low[x],dfn[y]);
}
}
#undef y
int main(){//freopen("test.in","r",stdin);//freopen("test.ans","w",stdout);
read(n),read(m);
for(register int i=,x,y;i<=m;++i)read(x),read(y),G.add(x,y);
fill(rb+,rb+n+,n+);
for(register int i=;i<=n;++i)if(!dfn[i])Top=,tarjan(i);
for(register int i=n-;i;--i)MIN(rb[i],rb[i+]);//dbg2(i,rb[i]);
for(register int i=;i<=n;++i)sum[i]=sum[i-]+rb[i];
read(q);for(register int i=,L,R,mid;i<=q;++i){
l=read(L),r=read(R);--L;
while(L<R){
mid=L+R+>>;
if(rb[mid]<=r)L=mid;
else R=mid-;
}//dbg(L);
printf("%I64d\n",(r-l+)*1ll*(r-l+)/-((L-l+)*1ll*(r+)-(sum[L]-sum[l-])));
}
return ;
}

总结:主要在问题二上卡住了。主要还是考虑从暴力入手,因为这个子区间问题基本思路就是固定左端点,算右端点,因为这题有两个单调的地方,所以可以用前缀和优化。

CF901C Bipartite Segments[点双+二分+前缀优化]的更多相关文章

  1. Codeforces 901C Bipartite Segments(Tarjan + 二分)

    题目链接  Bipartite Segments 题意  给出一个无偶环的图,现在有$q$个询问.求区间$[L, R]$中有多少个子区间$[l, r]$ 满足$L <= l <= r &l ...

  2. 【CF587D】Duff in Mafia 二分+前缀优化建图+2-SAT

    [CF587D]Duff in Mafia 题意:给你一张n个点m条边的无向图,边有颜色和边权.你要从中删去一些边,满足: 1.任意两条删掉的边没有公共的顶点.2.任意两条剩余的.颜色相同的边没有公共 ...

  3. CF901C. Bipartite Segments

    n<=300000,m<=300000的图,图上只有奇环,q<=300000个询问每次问:一个区间内有多少个子区间,满足只保留编号在该区间的点以及他们之间的边,可以构成一个二分图. ...

  4. Codeforces 901C Bipartite Segments

    Bipartite Segments 因为图中只存在奇数长度的环, 所以它是个只有奇数环的仙人掌, 每条边只属于一个环. 那么我们能把所有环给扣出来, 所以我们询问的区间不能包含每个环里的最大值和最小 ...

  5. 【CodeForces】901 C. Bipartite Segments

    [题目]C. Bipartite Segments [题意]给定n个点m条边的无向连通图,保证不存在偶数长度的简单环.每次询问区间[l,r]中包含多少子区间[x,y]满足只保留[x,y]之间的点和边构 ...

  6. POJ 3061 (二分+前缀和or尺取法)

    题目链接: http://poj.org/problem?id=3061 题目大意:找到最短的序列长度,使得序列元素和大于S. 解题思路: 两种思路. 一种是二分+前缀和.复杂度O(nlogn).有点 ...

  7. Codeforces Round #381 (Div. 2) D. Alyona and a tree 树上二分+前缀和思想

    题目链接: http://codeforces.com/contest/740/problem/D D. Alyona and a tree time limit per test2 secondsm ...

  8. Poj 3061 Subsequence(二分+前缀和)

    Subsequence Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 12333 Accepted: 5178 Descript ...

  9. Vijos1834 NOI2005 瑰丽华尔兹 动态规划 单调双端队列优化

    设dp[t][x][y]表示处理完前t个时间段,钢琴停留在(x,y)处,最多可以走多少个格子 转移时只需逆着当前倾斜的方向统计len个格子(len为时间区间的长度,len=t-s+1),如果遇到障碍就 ...

随机推荐

  1. 《The C Programming Language》学习笔记

    第五章:指针和数组 单目运算符的优先级均为2,且结合方向为自右向左. *ip++; // 将指针ip的值加1,然后获取指针ip所指向的数据的值 (*ip)++; // 将指针ip所指向的数据的值加1 ...

  2. [LuoguP1074]靶形数独_搜索

    靶形数独 题目链接:https://www.luogu.org/problem/P1074 数据范围:略. 题解: 传说中的大爆搜题啊. 我觉得这种题就是你能想到什么优化就直接上什么优化.... 这个 ...

  3. Kubernetes组件-CronJob(定时任务)

    ⒈简介: Kubernetes的Job资源在创建时会立即运行pod.但是许多批处理任务需要在特定的时间运行,或者在指定的时间间隔内重复运行.在Linux和类UNIX操作系统中,这些任务通常被称为cro ...

  4. Spring中@Component与@Bean的区别

    @Component和@Bean的目的是一样的,都是注册bean到Spring容器中. @Component  VS  @Bean @Component 和 它的子类型(@Controller, @S ...

  5. 怎样理解document的快捷方式属性

    所谓 "快捷方式属性" , 也就是说它们不是必须的, 只是在操作dom时可以更为方便地获取. 主要有下面8个: 1. 获取当前文档所属的window对象: document.def ...

  6. ztree入门

    ztree入门 ztree可用于权限管理,机构部门等有层次的数据 准备工作 ztree官网 点击右上角的GitHub或者码云的地址将demo下载到本地 在本地新建一个项目,将下载的文件中zTreeSt ...

  7. javaIO——AutoCloseable 小试

    前面在 IO 概述篇提到过,AutoCloseable 接口类会自动调用 close() 方法,那究竟具体怎么写呢?以及发生异常情况下或者多个资源是不是都能自动调用呢?我们来写一个简单的类测试一下就知 ...

  8. oc和swift对代码的分组,方便代码查找和导航用

    OC中对代码的分组: 在OC中对代码分组的命令是#pragma mark 来实现的, 分组用的是: #pragma mark 分组名 添加分割线: #pragma mark - 如果想添加分割线的同时 ...

  9. 转载:PHP扩展函数库-文件系统、进程与网络

    PHP的扩展函数库十分庞大,官方的非官方的,在这里只记录一些目前比较常用的扩展,对于这一部分,也只是记录其中一些核心的函数,不是一个全面记录.对于详细的扩展函数说明,需要在使用中参考PHP的用户手册. ...

  10. NSInvocation简单总结

    (1)用法 NSInvocation是调用函数的另一种方式,它将调用者,函数名,参数封装到一个对象,然后通过一个invoke函数来执行被调用的函数,其思想就是命令者模式,将请求封装成对象. 例如,有这 ...