注意到这个过程实质就是prim算法求最大生成树的过程。

首先通过离散化+线段树将$A+B$个点缩为上下各$O(n)$个点。

设已加入集合为$S$,未加入集合为$T$。

建立两棵线段树,维护所有在$T$集合中的点,以及从每个点连出去的边。

用一个大根堆维护所有横跨$ST$的边。每次取出堆顶的边,取出与这条边相连的任意一个属于$T$集合的点,若取不出则将这条边删去;否则将这个点加入$S$,并将所有连了它的边加入堆中。

时间复杂度$O(n\log n)$。

#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<queue>
#include<algorithm>
using namespace std;
typedef long long ll;
typedef pair<int,int>P;
const int N=100010,M=200010,T=525000,L=2000000;
int C,n,i,x,y,seq[M];ll ans;bool del[N];
struct E{int a,b,c,d,t;}e[N];
priority_queue<P>Q;
inline void read(int&a){char c;while(!(((c=getchar())>='0')&&(c<='9')));a=c-'0';while(((c=getchar())>='0')&&(c<='9'))(a*=10)+=c-'0';}
void NO(){
puts("-1");
exit(0);
}
inline void umax(int&a,int b){if(a<b)a=b;}
struct World{
int n,m,b[N<<1];
int val[T],g[T],v[L],nxt[L],ed;
inline int lower(int x){
int l=1,r=m,mid,t;
while(l<=r)if(b[mid=(l+r)>>1]<=x)l=(t=mid)+1;else r=mid-1;
return t;
}
inline void addx(int x,int y){b[++m]=x;b[++m]=y;}
void pre(){
b[++m]=1,b[++m]=n;
sort(b+1,b+m+1);
int _m=m;
m=0;
for(int i=1;i<=_m;i++)if(b[i]!=b[i-1])b[++m]=b[i];
}
inline void find(int&x,int&y){x=lower(x),y=lower(y);}
void cal(){
for(int i=1;i<m;i++)if(b[i+1]-b[i]-1){
if(!seq[i])NO();
ans+=1LL*seq[i]*(b[i+1]-b[i]-1);
}
}
void build(int x,int a,int b){
val[x]=b;
if(a==b)return;
int mid=(a+b)>>1;
build(x<<1,a,mid),build(x<<1|1,mid+1,b);
}
void ins(int x,int a,int b,int c,int d,int p){
if(c<=a&&b<=d){
v[++ed]=p;nxt[ed]=g[x];g[x]=ed;
return;
}
int mid=(a+b)>>1;
if(c<=mid)ins(x<<1,a,mid,c,d,p);
if(d>mid)ins(x<<1|1,mid+1,b,c,d,p);
}
int ask(int x,int a,int b,int c,int d){
if(c<=a&&b<=d)return val[x];
int mid=(a+b)>>1,t=0;
if(c<=mid)t=ask(x<<1,a,mid,c,d);
if(d>mid)umax(t,ask(x<<1|1,mid+1,b,c,d));
return t;
}
void change(int x,int a,int b,int c){
for(int&i=g[x];i;i=nxt[i]){
int j=v[i];
if(!del[j])Q.push(P(e[j].t,j)),del[j]=1;
}
if(a==b){val[x]=0;return;}
int mid=(a+b)>>1;
if(c<=mid)change(x<<1,a,mid,c);else change(x<<1|1,mid+1,b,c);
val[x]=max(val[x<<1],val[x<<1|1]);
}
}A,B;
int v[T];
void build(int x,int a,int b){
v[x]=0;
if(a==b)return;
int mid=(a+b)>>1;
build(x<<1,a,mid),build(x<<1|1,mid+1,b);
}
void change(int x,int a,int b,int c,int d,int p){
if(c<=a&&b<=d){umax(v[x],p);return;}
int mid=(a+b)>>1;
if(c<=mid)change(x<<1,a,mid,c,d,p);
if(d>mid)change(x<<1|1,mid+1,b,c,d,p);
}
void dfs(int x,int a,int b){
if(a==b){seq[a]=v[x];return;}
int mid=(a+b)>>1;
umax(v[x<<1],v[x]),umax(v[x<<1|1],v[x]);
dfs(x<<1,a,mid),dfs(x<<1|1,mid+1,b);
}
int main(){
read(A.n),read(B.n),read(C),read(n);
for(i=1;i<=n;i++){
read(e[i].a),read(e[i].b),read(e[i].c),read(e[i].d),read(e[i].t);
A.addx(e[i].a,e[i].b);
B.addx(e[i].c,e[i].d);
}
A.pre();
for(i=1;i<=n;i++)A.find(e[i].a,e[i].b);
build(1,1,A.m);
for(i=1;i<=n;i++)if(e[i].a<e[i].b)change(1,1,A.m,e[i].a,e[i].b-1,e[i].t);
dfs(1,1,A.m);
A.cal();
B.pre();
for(i=1;i<=n;i++)B.find(e[i].c,e[i].d);
build(1,1,B.m);
for(i=1;i<=n;i++)if(e[i].c<e[i].d)change(1,1,B.m,e[i].c,e[i].d-1,e[i].t);
dfs(1,1,B.m);
B.cal();
A.build(1,1,A.m);
B.build(1,1,B.m);
for(i=1;i<=n;i++){
A.ins(1,1,A.m,e[i].a,e[i].b,i);
B.ins(1,1,B.m,e[i].c,e[i].d,i);
}
A.change(1,1,A.m,1);
while(!Q.empty()){
x=Q.top().second;Q.pop();
if(y=A.ask(1,1,A.m,e[x].a,e[x].b)){
ans+=e[x].t;
A.change(1,1,A.m,y);
Q.push(P(e[x].t,x));
continue;
}
if(y=B.ask(1,1,B.m,e[x].c,e[x].d)){
ans+=e[x].t;
B.change(1,1,B.m,y);
Q.push(P(e[x].t,x));
continue;
}
}
if(A.val[1]||B.val[1])NO();
return printf("%lld",ans),0;
}

  

BZOJ4388 : JOI2012 invitation的更多相关文章

  1. 【BZOJ4388】JOI2012 invitation 堆+线段树+并查集(模拟Prim)

    [BZOJ4388]JOI2012 invitation Description 澳洲猴举办了一场宴会,他想要邀请A个男生和B个女生参加,这A个男生从1到A编号,女生也从1到B编号.现在澳洲猴知道n组 ...

  2. 【JOISC2012 / bzoj4388】Invitation

    Description link Solution 可以发现题目在模拟 Prim 算法求最大生成树的过程,树边故答案与起点 \(C\) 无关. 先把所有区间离散化,注意对于一个区间 \([l,r]\) ...

  3. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  4. BZOJ 4388 [JOI2012春季合宿]Invitation (线段树、二叉堆、最小生成树)

    题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=4388 题解 模拟Prim算法? 原题所述的过程就是Prim算法求最大生成树的过程.于是我 ...

  5. HDU 1535 Invitation Cards(最短路 spfa)

    题目链接: 传送门 Invitation Cards Time Limit: 5000MS     Memory Limit: 32768 K Description In the age of te ...

  6. POJ 1511 Invitation Cards (spfa的邻接表)

    Invitation Cards Time Limit : 16000/8000ms (Java/Other)   Memory Limit : 524288/262144K (Java/Other) ...

  7. POJ 1511 Invitation Cards (最短路spfa)

    Invitation Cards 题目链接: http://acm.hust.edu.cn/vjudge/contest/122685#problem/J Description In the age ...

  8. HDU1535——Invitation Cards(最短路径:SPAF算法+dijkstra算法)

    Invitation Cards DescriptionIn the age of television, not many people attend theater performances. A ...

  9. Poj 1511 Invitation Cards(spfa)

    Invitation Cards Time Limit: 8000MS Memory Limit: 262144K Total Submissions: 24460 Accepted: 8091 De ...

随机推荐

  1. python4delphi import lxml pandas 出错的小结

    环境: 1.win10 64位 2.delphi xe8 3.python2.7 4.python4delphi  (svn 2015-03-21 发布的83版本号) 5.lxml 3.4.4(通过p ...

  2. 常州Day4题解

    1. 高精度 这题略水,字符串可过,还不加压位等,操作只有BitShift和add/sub,不过编程复杂度有些高.(输出都是二进制我能说些什么...) 2. N皇后问题 (警告! 不是平时你见到的N皇 ...

  3. 【云计算】Kubernetes、Marathon等框架需要解决什么样的问题?

    闲谈Kubernetes 的主要特性和经验分享       Capitalonline全球云主机.全球私有网络,免费试用进行时 »   主要介绍 Kubernetes 的主要特性和一些经验.先从整体上 ...

  4. Android 中的Force Close

    今天写程序时遇到一个问题,领导希望在点击了setting里的force close 后,程序依然能够响应以前用alarmManager注册的receiver. 在网上看到了一些文章,写的是如何建立一个 ...

  5. Linux下挂载NTFS格式的U盘或硬盘

    我们知道在Linux下挂载fat32的U盘非常容易,使用mount /dev/drive_name /mnt/指定目录这样就可以挂载了,但是如果U盘或者硬盘的格式是NTFS的话,那么Linux是不能识 ...

  6. 通过eclipse配置Spring MVC项目

    上一篇刚建立了一个简单的Spring项目,其实Spring MVC是一个和Struts2一样的基于MVC设计模式的web框架,并且继承了MVC的优点,是基于请求驱动的轻量级的web框架,spring ...

  7. 【动态规划】拦截导弹_dilworth定理_最长递增子序列

    问题 K: [动态规划]拦截导弹 时间限制: 1 Sec  内存限制: 256 MB提交: 39  解决: 10[提交][状态][讨论版] 题目描述 张琪曼:“老师,修罗场是什么?” 墨老师:“修罗是 ...

  8. Heap:Sunscreen(POJ 3614)

    晒太阳 题目大意:一堆牛,为了避免晒太阳会灼烧自己,然后他们自己有自己的防晒指数(一个区间),防晒霜可以提高防晒因数SPF,大了不行小了不行,现在有一桶防晒霜,他们提供一定的SPF,但是最多可以提供k ...

  9. 【好用的小技巧】win8兼容、网页不让复制

    1.今天下了个matlab7,我用的是win8系统,不兼容. 解决:鼠标右键matlab7的快捷键,点击属性,选择兼容性,选择window vista即可运行 2.在一个 网页上看到一个对自己很有帮助 ...

  10. [] ubuntu 14.04 搜狗拼音输入法安装

    1.需要现在ubuntu下安装fcitx和libssh2-1命令如下 sudo apt-get install fcitx sudo apt-get install libssh2-1 注意最好是先卸 ...