LINK:VMware和基站

一道 做法并不常见的题目 看起来很难写 其实set维护线段就可以解决了。

容易想到 第二个操作借用启发式合并可以得到一个很不错的复杂度 不过利用线段树维护这个东西 在区间覆盖的时候并不能很好的维护。

一个想法是 分块 不过操作比较ex.

第一个操作和第二个操作连在一起会非常的难以处理。

这里给出的做法是:使用set来维护整段的线段 这样在第一个区间覆盖的情况下 每次区间至多增加两个或者可能减少.

在第一步中就是logn的时间内可以维护了.不过细节较多我分类讨论了好几种情况。

考虑第二步 不妨每个基站都开一个set 利用启发式合并来做。

不过 不存在操作1的时候启发式合并 的总复杂度为nlog^2 存在操作1的时候 容易发现 每次区间最多增加两个 那么均摊到启发式合并上也是Qlog的时间。

对于第三步 直接在set中二分 分类讨论在左边还是右边即可.

挺难写的 三个函数都出了锅 第一个是少考虑了情况 第二个是 没有对s进行修改 第三个则是答案的情况存在小细节问题 写死我了.

const int MAXN=100010,G=3;
int n,Q;
struct wy
{
int l,r,id;
wy(int x,int y,int z){l=x;r=y;id=z;}
inline bool operator <(wy a)const {return l<a.l;}
};
set<wy>s;set<pii>g[MAXN];
set<wy>::iterator it;
set<pii>::iterator itt;
char a[10];
int f[MAXN];
inline void cover(int l,int r,int x)
{
it=s.upper_bound(wy(l,r,x));--it;
if((*it).l!=l)
{
int id=(*it).id;int L=(*it).l;int R=(*it).r;
g[id].erase(mk(L,R));g[id].insert(mk(L,l-1));
s.erase(it);s.insert(wy(L,l-1,id));
g[x].insert(mk(l,r));
s.insert(wy(l,r,x));it=s.find(wy(l,r,x));
if(R>r)
{
g[id].insert(mk(r+1,R));
s.insert(wy(r+1,R,id));
return;
}
if(R==r)return;++it;
while(it!=s.end())
{
R=(*it).r;
if(R<=r)
{
g[(*it).id].erase(mk((*it).l,(*it).r));
s.erase(it);it=s.find(wy(l,r,x));++it;
if(R==r)break;
}
else
{
id=(*it).id;
g[id].erase(mk((*it).l,R));
g[id].insert(mk(r+1,R));
s.erase(it);s.insert(wy(r+1,R,id));
break;
}
}
}
else
{
int id=(*it).id;int R=(*it).r;
//cout<<id<<endl;
g[id].erase(mk(l,(*it).r));
s.erase(it);s.insert(wy(l,r,x));it=s.find(wy(l,r,x));
g[x].insert(mk(l,r));
if(R>r)
{
g[id].insert(mk(r+1,R));
s.insert(wy(r+1,R,id));
return;
}
if(R==r)return;++it;
while(it!=s.end())
{
R=(*it).r;
//cout<<R<<endl;
if(R<=r)
{
g[(*it).id].erase(mk((*it).l,(*it).r));
s.erase(it);it=s.find(wy(l,r,x));++it;
if(R==r)break;
}
else
{
id=(*it).id;
g[id].erase(mk((*it).l,R));
g[id].insert(mk(r+1,R));
s.erase(it);s.insert(wy(r+1,R,id));
break;
}
}
}
}
inline void replace(int &x,int &y)
{
if(x==y)return;
if(g[x].size()>g[y].size())swap(x,y);
itt=g[x].begin();
while(itt!=g[x].end())
{
it=s.find(wy((*itt).F,(*itt).S,x));
s.erase(it);
s.insert(wy((*itt).F,(*itt).S,y));
g[y].insert(*itt);
++itt;
}
g[x].clear();
}
inline int ask(int D,int L,int R,int id)
{
if(!g[id].size())return -1;
int ans=-1;
itt=g[id].lower_bound(mk(L,0));
if(itt!=g[id].end()&&(*itt).F<=R)ans=max(ans,abs(D-(*itt).F));
if(itt!=g[id].begin())
{
--itt;
//cout<<(*itt).F<<' '<<(*itt).S<<endl;
if((*itt).F<=L&&(*itt).S>=L)ans=max(ans,abs(D-L));
}
itt=g[id].upper_bound(mk(R,n+1));
if(itt!=g[id].begin())
{
--itt;
if((*itt).F<=R&&(*itt).S>=R)ans=max(ans,abs(D-R));
if((*itt).S<=R&&(*itt).S>=L)ans=max(ans,abs(D-(*itt).S));
//cout<<(*itt).S<<' '<<(*itt).F<<endl;
}
return ans;
}
int main()
{
//freopen("1.in","r",stdin);
gt(n);gt(Q);
rep(1,n,i)g[i].insert(mk(i,i)),s.insert(wy(i,i,i)),f[i]=i;
rep(1,Q,i)
{
int x,y,z;
gc(a);gt(x);gt(y);
if(a[1]=='c')gt(z),cover(x,y,f[z]);
if(a[1]=='r')replace(f[x],f[y]);
if(a[1]=='f')gt(z),put(ask(x,max(x-y,1),min(x+y,n),f[z]));
}
return 0;
}

牛客挑战赛40 VMware和基站 set 二分 启发式合并 区间覆盖的更多相关文章

  1. 5.15 牛客挑战赛40 C 小V和字符串 数位dp 计数问题

    LINK:小V和字符串 容易想到只有1个数相同的 才能有贡献. 知道两个01串 那么容易得到最小步数 大体上就是 第一个串的最前的1和第二个串最前的1进行匹配. 容易想到设f[i][j]表示 前i位1 ...

  2. 5.15 牛客挑战赛40 E 小V和gcd树 树链剖分 主席树 树状数组 根号分治

    LINK:小V和gcd树 时限是8s 所以当时好多nq的暴力都能跑过. 考虑每次询问暴力 跳父亲 这样是nq的 4e8左右 随便过. 不过每次跳到某个点的时候需要得到边权 如果直接暴力gcd的话 nq ...

  3. 5.15 牛客挑战赛40 B 小V的序列 关于随机均摊分析 二进制

    LINK:小V的序列 考试的时候 没想到正解 于是自闭. 题意很简单 就是 给出一个序列a 每次询问一个x 问序列中是否存在y 使得x^y的二进制位位1的个数<=3. 容易想到 暴力枚举. 第一 ...

  4. 牛客挑战赛 39 牛牛与序列 隔板法 容斥 dp

    LINK:牛牛与序列 (牛客div1的E题怎么这么水... 还没D难. 定义一个序列合法 当且仅当存在一个位置i满足 $a_i>a_,a_j<a_$且对于所有的位置i,$1 \leq a_ ...

  5. 牛客练习赛16 F 选值【二分/计数】

    链接:https://www.nowcoder.com/acm/contest/84/F 来源:牛客网 题目描述 给定n个数,从中选出三个数,使得最大的那个减最小的那个的值小于等于d,问有多少种选法. ...

  6. 牛客练习赛34 little w and Segment Coverage (差分区间)

    链接:https://ac.nowcoder.com/acm/contest/297/C来源:牛客网 题目描述 小w有m条线段,编号为1到m. 用这些线段覆盖数轴上的n个点,编号为1到n. 第i条线段 ...

  7. 牛客小白月赛16 小石的妹子 二分 or 线段树

    牛客小白月赛16 这个题目我AC之后看了一下别人的题解,基本上都是线段树,不过二分也可以. 这个题目很自然就肯定要对其中一个进行排序,排完序之后再处理另外一边,另一边记得离散化. 怎么处理呢,你仔细想 ...

  8. 牛客挑战赛 30 A 小G数数

    题目链接:https://ac.nowcoder.com/acm/contest/375/A 分析:我写的时候竟然把它当成了DP....... 还建了个结构体DP数组,保存一二位,不知道当时脑子在抽啥 ...

  9. 良心送分题(牛客挑战赛35E+虚树+最短路)

    目录 题目链接 题意 思路 代码 题目链接 传送门 题意 给你一棵树,然后把这棵树复制\(k\)次,然后再添加\(m\)条边,然后给你起点和终点,问你起点到终点的最短路. 思路 由于将树复制\(k\) ...

随机推荐

  1. 洛谷 P6082 [JSOI2015]salesman

    题意 给定一棵\(n\)个点的树,有点权,你从\(1\)号点开始一次旅行,最后回到\(1\)号点.每到达一个点,你就能获得等于该点点权的收益, 但每个点都有进入该点的次数限制,且每个点的收益只能获得一 ...

  2. 如何科学地完成一场 AR 发布会?全在这份超细节活动策划 Xmind 里了

    你们在哪个酒店搭的景? 5 月 28 日,网易智慧企业完成了一场实景人物拍摄 + 虚拟舞台渲染的 AR 线上见面会.非常有趣的是,在直播过程中,不止一位观众问我们,“你们是在哪个酒店搭的景?”.看来我 ...

  3. requests库入门笔记1

    1.使用requests库发送请求,fiddler无法抓到包:使用浏览器请求相同的url,可以抓到包 在请求参数中添加 proxies参数,如下: proxies = { 'http': 'http: ...

  4. Python split分割字符串

    s = input(); str = s.split("-") print("{}+{}".format(str[0],str[-1]))

  5. 数据可视化之powerBI技巧(二十二)利用这个方法,帮你搞定Power BI"增量刷新"

    Power BI的增量刷新功能现在已经对Pro用户开通,但由于种种限制,很多人依然无法使用无这个功能,所以,每一次刷新,都要彻底更新数据集.这对于量级比较大的数据集来说,着实是一件耗费时间的事情. 拿 ...

  6. 数据可视化之powerBI技巧(十一)基于SQL思维的PowerBI DAX实战

    本文来自于PowerBI星球嘉宾天行老师的分享,天行老师不仅DAX使用娴熟,更是精通SQL,下面就来欣赏他利用SQL思维编写DAX解决问题的一个实战案例. 基于SQL思维使用DAX解决实战问题 作者: ...

  7. 03-Django模型知识1

    模型:是python中的类对应数据库中的表. ORM:对象关系映射 1.基础示例 学生类模型字段 class blog(models.Model): title = models.CharField( ...

  8. 在 CentOS 7(Linux)上部署ASP.NET Core 2.2 Web应用程序(Tengine、Asp.Net Core MVC、Centos 7、MySql)

    一.前言 1.简单记录一下Linux CentOS 7中安装与配置Tengine的详细步骤. 2.简单比较一下Tengine 和Nginx 3.搭建Asp.net Core和部署 Web程序 4.总结 ...

  9. 【原创】linux设备模型之kset/kobj/ktype分析

    背 景 Read the fucking source code! --By 鲁迅 A picture is worth a thousand words. --By 高尔基 说明: Kernel版本 ...

  10. 解决nginx在Linux中已经正常启动,Windows端的浏览器却无法访问的问题

    一:查看Linux中nginx已经正常启动 二:查看80端口,未被占用 三:检查防火墙的问题 关闭防火墙:chkconfig iptables off //失败 暂时关闭防火墙:service ipt ...