UOJ356 [JOI2017春季合宿] Port Facility 【启发式合并】【堆】【并查集】
题目分析:
好像跑得很快,似乎我是第一个启发式合并的。
把玩具看成区间。首先很显然如果有两个玩具的进出时间有$l1<l2<r1<r2$的关系,那么这两个玩具一定在不同的栈中间。
现在假设一定有解,我们怎么得到答案呢?排序会使得计算变得方便,下面我们按照左端点排序。
想象一条扫描线,从左往右,当它遇到了一个区间的左端点的时候,我们尝试着将原先不在一起的合并,所有和这个不同栈的都被合并。
我们可以想象一个并查集,使用堆维护并查集。堆内存储并查集内元素的右端点。在最外面再用一个大堆来存储每个并查集的右端最小值(堆套堆)
实际上在每个并查集唯一要考虑的是在当前扫描线右端的最小的点的位置。若它不在当前玩具控制的区间内则该并查集一定不被该玩具影响。
否则将它提取出来,然后对并查集启发式合并。不仅如此,当前考虑的玩具也被合并在内。
现在考虑无解。无解实际上是冲突,如果有两个互异玩具集合,如果新出现一个玩具和其它两个不兼容则无解。
在上面的基础上,我们要做的只有把一个并查集再分裂成两个部分,然后两个部分分开合并即可。
时间复杂度$O(nlogn)$
#include<bits/stdc++.h>
using namespace std;
typedef pair<int,int> pr; const int maxn = (<<)+;
const int mod = ;
struct line{int l,r,num;}A[maxn];
int cnt,n,im[maxn]; void read(){
scanf("%d",&n);
for(int i=;i<=n;i++)scanf("%d%d",&A[i].l,&A[i].r),A[i].num=i;
sort(A+,A+n+,[](line a,line b){return a.l < b.l;});
} priority_queue<pr,vector<pr>,greater<pr> > pq;
priority_queue<int,vector<int>,greater<int> > s[maxn>>],t[maxn>>];
int um,rs[maxn],num;
void work(){
for(int i=;i<=n;i++) s[i].push(A[i].r);
for(int i=;i<=n;i++){
while(!pq.empty()){
pr k = pq.top(); if(k.first > A[i].l) break; pq.pop();
if(s[k.second].size() && s[k.second].top() == k.first) s[k.second].pop();
else t[k.second].pop();
if(!s[k.second].size() && !t[k.second].size()) cnt++;
else{
if(!t[k.second].size()) {pq.push(make_pair(s[k.second].top(),k.second));continue;}
if(!s[k.second].size()) {pq.push(make_pair(t[k.second].top(),k.second));continue;}
if(s[k.second].size() && t[k.second].size())
pq.push(make_pair(min(s[k.second].top(),t[k.second].top()),k.second));
}
}
if(pq.empty()){pq.push(make_pair(A[i].r,i));continue;}
um = i;int place = ;num = ;
while(!pq.empty()){pr k = pq.top();if(k.first < A[i].r) {rs[++num] = k.second;pq.pop();}else break;}
for(int j=;j<=num;j++){
int zeta = rs[j];
if(s[zeta].size() && t[zeta].size() && s[zeta].top() < A[i].r && t[zeta].top() < A[i].r){
puts("");return;
}
int sum1 = s[um].size() + t[um].size(),sum2 = s[zeta].size()+t[zeta].size();
if(sum1 > sum2){
if((s[zeta].size() && s[zeta].top() < A[i].r) ^ place){
while(t[zeta].size()){s[um].push(t[zeta].top());t[zeta].pop();}
while(s[zeta].size()){t[um].push(s[zeta].top());s[zeta].pop();}
}
else{
while(s[zeta].size()){s[um].push(s[zeta].top());s[zeta].pop();}
while(t[zeta].size()){t[um].push(t[zeta].top());t[zeta].pop();}
}
}else{
if((s[zeta].size() && s[zeta].top() < A[i].r) ^ place){
while(t[um].size()){s[zeta].push(t[um].top());t[um].pop();}
while(s[um].size()){t[zeta].push(s[um].top());s[um].pop();}
place ^= ;
}else{
while(s[um].size()){s[zeta].push(s[um].top());s[um].pop();}
while(t[um].size()){t[zeta].push(t[um].top());t[um].pop();}
}
um = zeta;
}
}
if(!t[um].size())pq.push(make_pair(s[um].top(),um));
else if(!s[um].size()) pq.push(make_pair(t[um].top(),um));
else pq.push(make_pair(min(s[um].top(),t[um].top()),um));
}
while(!pq.empty()){cnt++;pq.pop();}
int ans = ;while(cnt--){ans*=;ans%=mod;}
printf("%d",ans);
} int main(){
read();
work();
return ;
}
UOJ356 [JOI2017春季合宿] Port Facility 【启发式合并】【堆】【并查集】的更多相关文章
- [JOI2017春季合宿]Port Facility[set、二分图]
题意 你有两个栈,有 \(n\) 个货物,每个货物有一个进栈时间和出栈时间(所有时间的并集是1~2n),问有多少种不同的入栈方案. \(n\le 10^6\) 分析 把每个货物的存在看成区间,相交的区 ...
- BZOJ 2733 [HNOI2012]永无乡(启发式合并+Treap+并查集)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2733 [题目大意] 给出n个点,每个点都有自己的重要度,现在有连边操作和查询操作, 查 ...
- UOJ356 【JOI2017春季合宿】Port Facility
暴力就是O(n^2)连边,二分图,这样只有22分. 我们考虑优化建边,我们按照左端点排序,对于一个新加进来的线段,我们向左端点距其最近的和他相交的线段连边,别的相交的我们连同色边,当一个点连了两条同色 ...
- UOJ #356. 【JOI2017春季合宿】Port Facility
Description 小M有两个本质不同的栈. 无聊的小M找来了n个玩具.之后小M把这n个玩具随机顺序加入某一个栈或把他们弹出. 现在小M告诉你每个玩具的入栈和出栈时间,现在她想考考小S,有多少种方 ...
- 【JOI2017春季合宿】Port Facility
http://uoj.ac/problem/356 题解 思路和\(NOIP\)双栈排序差不多. 对于两个元素,若\(l_1<l_2<r_1<r_2\)那么它们不能在一个栈里,我们连 ...
- UOJ #357. 【JOI2017春季合宿】Sparklers
Description 小S和小M去看花火大会. 一共有 n 个人按顺序排成一排,每个人手上有一个仅能被点燃一次的烟花.最开始时第 K 个人手上的烟花是点燃的. 烟花最多能燃烧 T 时间.每当两个人的 ...
- JOI2017 春季合宿:Railway Trip
自己的AC做法似乎离正解偏了十万八千里而且复杂了不少--不管怎样还是记录下来吧. 题意: 题目链接: JOISC2017 F - AtCoder JOISC2017 F - LOJ \(N\)个车站排 ...
- BZOJ 4388 [JOI2012春季合宿]Invitation (线段树、二叉堆、最小生成树)
题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=4388 题解 模拟Prim算法? 原题所述的过程就是Prim算法求最大生成树的过程.于是我 ...
- BZOJ 4221 [JOI2012春季合宿]Kangaroo (DP)
题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=4221 题解 orz WYC 爆切神仙DP 首先将所有袋鼠按大小排序.考虑从前往后DP, ...
随机推荐
- Jmeter实例(二)简单的性能测试场景
我们在性能测试过程中,首先应该去设计测试场景,模拟真实业务发生的情境,然后针对这些场景去设计测试脚本.为了暴露出性能问题,要尽可能的去模拟被测对象可能存在瓶颈的测试场景. 我在本地部署了一个项目,可以 ...
- [loadrunner]通过检查点判定事务是否成功
//检查点设置语句 //tmp在此时为临时参数 web_reg_find("SaveCount=tmp", "Text=xxx", LAST); ...
- Python文本处理
文本处理 (一)对文本操作的流程: 打开文件,得到文件句柄并赋值给一个变量 通过句柄对文件进行操作 关闭文件 open(file, mode='r', buffering=None, encoding ...
- Maven安装与环境配置(Windows)
1.下载安装包 在Maven官网下载最新版的安装包:http://maven.apache.org/download.cgi 2.解压安装包 3.配置Maven环境变量 配置M2_HOME环境变量,指 ...
- 关于js作用域问题
补充: function Foo(name,age){ this.name=name; this.age=age; this.getName=function(){ console.log(this) ...
- 爬虫——scrapy框架
Scrapy是一个异步处理框架,是纯Python实现的爬虫框架,其架构清晰,模块之间的耦合程度低,可拓展性强,可以灵活完成各种需求.我们只需要定制几个模块就可以轻松实现一个爬虫. 1.架构 Scra ...
- winform启动界面+登录窗口
需求场景:先展示启动界面,然后打开登录界面,如果登录成功就跳转到主界面 首先在程序的入口路径加载启动界面,使用ShowDialog显示界面, 然后在启动界面中添加定时器,来实现显示一段时间的效果,等到 ...
- asp.net mvc Dateset读取Excel数据
//处理Excel //读取Excel [NonAction] public static DataSet ExcelToDS(string Path) { //根据情况选择合适的连接字符,参考msd ...
- hangfire使用笔记
1.导入nuget包 2.配置 简单配置后就可以写自己的Job了 注意:Hangfire.RecurringJobExtensions这个扩展支持两种job添加方法:json配置文件和特性.但由于时区 ...
- js的日期操作:String转date日期格式、求日期差
一.在js中String类型转成date格式 var date = new Date("2018-9-21 14:58:43");//就是这么简单 二.date转String类型就 ...