www.lydsy.com/JudgeOnline/upload/task.pdf

第一题题意可以转为选一个长度k的序列,每一项二进制的1的位置被下一项包含,且总和为1,考虑每个二进制位的出现位置,可以转化为一个多重背包求方案数的问题。

第二题构成一些环,可以每个环直接计算,然后合并答案。

第三题区间包含相当于单向边,区间相交不包含就是双向边,将区间相交不包含的情况缩点,剩余的单向边构成一棵树(几个被缩起来的区间可以用它们的并集表示),用lct维护树形态,查询即为询问两点是否是祖先关系,对于修改,由于区间长度递增,只有至多两种缩点情况:1.新加入的区间与某棵树的根缩点(另用平衡树维护所有根对应的区间,实际上需要再分几类);2.新加入的区间与树上非根的点缩点(另用线段树维护所有非根节点对应的区间)。

前两题代码实现比较简单,所以这里只给出第三题的代码。

#include<bits/stdc++.h>
int _(){
int x=,f=,c=getchar();
while(c<)c=='-'?f=-:,c=getchar();
while(c>)x=x*+c-,c=getchar();
return x*f;
}
#define lc ch][0
#define rc ch][1
#define fa ch][2
const int N=2e5+,inf=0x3f3f3f3f;
struct itv{
int l,r,id;
bool operator<(const itv&w)const{return l<w.l;}
};
struct cmpl{bool operator()(const itv&a,const itv&b){return a.l>b.l;}};
struct cmpr{bool operator()(const itv&a,const itv&b){return a.r<b.r;}};
std::set<itv>st;
int n,f[N],ch[N][],idp=;
int qs[N][],xs[N],xp=,mx=,tl[],tr[];
std::priority_queue<itv,std::vector<itv>,cmpl>ls[N];
std::priority_queue<itv,std::vector<itv>,cmpr>rs[N];
int gf(int x){
while(x!=x[f])x=x[f]=x[f][f];
return x;
}
bool nrt(int x){return x==x[fa][lc]||x==x[fa][rc];}
int wc(int x){return x==x[fa][rc];}
void rot(int x){
int f=x[fa],g=f[fa],d=wc(x);
if(nrt(f))g[ch][wc(f)]=x;
x[fa]=g;
(f[ch][d]=x[ch][d^])[fa]=f;
(x[ch][d^]=f)[fa]=x;
}
void sp(int x){
while(nrt(x)){
int f=x[fa];
if(nrt(f))rot(wc(x)==wc(f)?f:x);
rot(x);
}
}
int acs(int x){int y=;for(;x;sp(x),x[rc]=y,y=x,x=x[fa]);return y;}
void lk(int a,int b){acs(a)[fa]=b;}
void ctlk(int a,int b){acs(a),sp(a);a[lc][fa]=,a[lc]=,a[fa]=b;}
int max(int a,int b){return a>b?a:b;}
int min(int a,int b){return a<b?a:b;}
void tl_up(int x){
tl[x+mx]=ls[x].size()?ls[x].top().l:inf;
for(x=x+mx>>;x;x>>=)tl[x]=min(tl[x<<],tl[(x<<)+]);
}
void tr_up(int x){
tr[x+mx]=rs[x].size()?rs[x].top().r:-inf;
for(x=x+mx>>;x;x>>=)tr[x]=max(tr[x<<],tr[(x<<)+]);
}
int $(int x){
return std::lower_bound(xs+,xs+xp+,x)-xs;
}
void tr_ins(itv w){
int _l=$(w.l),_r=$(w.r);
ls[_r].push(w),tl_up(_r);
rs[_l].push(w),tr_up(_l);
}
void chk(itv w,int z){
int p,u;
while(tl[z]<w.l){
for(p=z;p<mx;p<<=,p+=tl[p]>=w.l);
p-=mx;
u=ls[p].top().id;
if(u==f[u])ctlk(u,f[u]=w.id);
ls[p].pop();
tl_up(p);
}
while(tr[z]>w.r){
for(p=z;p<mx;p<<=,p+=tr[p]<=w.r);
p-=mx;
u=rs[p].top().id;
if(u==f[u])ctlk(u,f[u]=w.id);
rs[p].pop();
tr_up(p);
}
}
void tr_del(itv w){
for(int l=$(w.l)+mx,r=$(w.r)+mx;r-l>;l>>=,r>>=){
if(~l&)chk(w,l+);
if(r&)chk(w,r-);
}
}
void ins(itv w){
tr_del(w);
std::set<itv>::iterator it=st.lower_bound(w);
if(it!=st.end()&&it->l==w.l&&it->r>w.r)return lk(w.id,f[w.id]=it->id);
if(it!=st.begin()&&(--it)->r>w.l){
int u=it->id;
if(it->r>=w.r)return lk(w.id,f[w.id]=u);
w.l=it->l,f[u]=w.id;
lk(u,w.id);
st.erase(it);
}
while((it=st.lower_bound(w))!=st.end()&&it->l<w.r){
int u=it->id;
if(it->r>w.r)w.r=it->r,f[u]=w.id;
else tr_ins(*it);
lk(u,w.id);
st.erase(it);
}
st.insert(w);
}
bool query(int x,int y){
x=gf(x),y=gf(y);
if(x==y)return ;
x=acs(x);
int y0=y;
for(;nrt(y);y=y[fa]);
sp(y0);
return x==y;
}
int main(){
n=_();
for(int i=;i<=n;++i){
qs[i][]=_();
qs[i][]=_();
qs[i][]=_();
if(qs[i][]==)xs[++xp]=qs[i][],xs[++xp]=qs[i][];
}
std::sort(xs+,xs+xp+);
for(;mx<=xp+;mx<<=);
for(int i=mx*-;i;--i)tl[i]=inf,tr[i]=-inf;
for(int i=;i<=n;++i){
f[i]=i;
int o=qs[i][],x=qs[i][],y=qs[i][];
if(o==)ins((itv){x,y,++idp});
else puts(query(x,y)?"YES":"NO");
}
return ;
}

bzoj5003: 与链 5004: 开锁魔法II 5005:乒乓游戏的更多相关文章

  1. BZOJ 5004: 开锁魔法II 期望 + 组合

    Description 题面:www.lydsy.com/JudgeOnline/upload/task.pdf Input Output 一般概率题有两种套路: 满足条件的方案/总方案. 直接求概率 ...

  2. BZOJ 5004: 开锁魔法II

    比较显然 #include<cstdio> #include<algorithm> #include<cstring> using namespace std; i ...

  3. 【bzoj5004】开锁魔法II 组合数学+概率dp

    题目描述 有 $n$ 个箱子,每个箱子里有且仅有一把钥匙,每个箱子有且仅有一把钥匙可以将其打开.现在随机打开 $m$ 个箱子,求能够将所有箱子打开的概率. 题解 组合数学+概率dp 题目约定了每个点的 ...

  4. hrb——开锁魔法I——————【规律】

    解题思路:从1到n的倒数之和. #include<stdio.h> #include<string.h> #include<algorithm> using nam ...

  5. hihocoder1075【开锁魔法】

    hihocoder1075[开锁魔法] 题意是给你一个 \(1-n\) 的置换,求选 \(k\) 个可以遍历所有点的概率. 题目可以换个模型:有 \(n\) 个球,有 \(cnt\) 种不同的颜色,求 ...

  6. hihocoder 1075 : 开锁魔法III

    描述 一日,崔克茜来到小马镇表演魔法. 其中有一个节目是开锁咒:舞台上有 n 个盒子,每个盒子中有一把钥匙,对于每个盒子而言有且仅有一把钥匙能打开它.初始时,崔克茜将会随机地选择 k 个盒子用魔法将它 ...

  7. #1075 : 开锁魔法III

    描述 一日,崔克茜来到小马镇表演魔法. 其中有一个节目是开锁咒:舞台上有 n 个盒子,每个盒子中有一把钥匙,对于每个盒子而言有且仅有一把钥匙能打开它.初始时,崔克茜将会随机地选择 k 个盒子用魔法将它 ...

  8. Hiho #1075: 开锁魔法III

    Problem Statement 描述 一日,崔克茜来到小马镇表演魔法. 其中有一个节目是开锁咒:舞台上有 n 个盒子,每个盒子中有一把钥匙,对于每个盒子而言有且仅有一把钥匙能打开它.初始时,崔克茜 ...

  9. HihoCoder 1075 开锁魔法III(概率DP+组合)

    描述 一日,崔克茜来到小马镇表演魔法. 其中有一个节目是开锁咒:舞台上有 n 个盒子,每个盒子中有一把钥匙,对于每个盒子而言有且仅有一把钥匙能打开它.初始时,崔克茜将会随机地选择 k 个盒子用魔法将它 ...

随机推荐

  1. day01计算机组成与操作系统

    1.什么是编程语言编程语言是程序员与计算机之间的沟通介质 2.什么是编程编程的过程就是程序员通过某种语言将命令给到计算机并让计算机表达出来 修改后:编程就是程序员按照某种语法规则将自己想让计算机做的事 ...

  2. 2017-2018-1 20155219《信息安全技术》实验二——Windows口令破解

    2017-2018-1 20155320<信息安全技术>实验二--Windows口令破解 实验目的 了解Windows口令破解原理 对信息安全有直观感性认识 能够运用工具实现口令破解 实验 ...

  3. Is this Sentence Difficult? Do you Agree? -paper

    术语: 数据lemma:词根,词元(词的基本形式,如名词单数或动词的不定式形式) content words:实词part-of-speech:词性object-relative clauses an ...

  4. java-类中需注意的问题

    1.对成员变量的操作只能放在方法中,方法可以对成员变量和该方法中声明的局部变量进行操作. 在声明类的成员变量时,可以同时赋予初值,例如: class Test { int a=12; float b= ...

  5. python django day 4 database

    django-admin.py startproject learn_models # 新建一个项目 cd learn_models # 进入到该项目的文件夹 django-admin.py star ...

  6. Python知识点整理,基础5 - 文件操作

  7. C语言指针入门

    指针指包含地址 声明指针: int num:声明整数 int *num:声明指针,下面的声明也是等价的,空格的使用指示个人爱好 int* num; int  *  num; int  *num; in ...

  8. MySQL Innodb Engine --独立表空间参数(innodb_file_per_table)

    MySQL中参数innodb_file_per_table决定将表存放于ibdata*的共享表空间还是独立的.ibd文件的独立表空间. ================================ ...

  9. 【转载】Win10桌面图标有小箭头怎么去掉?Win10去掉桌面图标小箭头的方法

    以下文章转载至系统之家 网址:http://www.xitongzhijia.net/xtjc/20190104/146560.html Win10桌面图标有小箭头怎么去掉?Win10去掉桌面图标小箭 ...

  10. 扩大UIButton的选区

          设计为了美观.button有时设计的太小,这么一来是好看了.可是button不好点中了.曾经有非常多解决方法,如用hitTest等方法. 都比較复杂.后来我发现一个简单的方法. 原理就是U ...