【CF833E】Caramel Clouds(线段树)
【CF833E】Caramel Clouds(线段树)
题面
题解
首先把区间一段一段分出来,那么只有四种情况。
要么没有被任何一朵云被覆盖,那么直接就会产生这一段的贡献。
要么被一朵云覆盖,要么被两朵云覆盖。
要么被三朵及以上的云所覆盖,那么这段的贡献永远取不到。
对于每朵云预处理出只被其覆盖的区间长度\(len[i]\),这样子就能处理只选择一朵云的贡献了。
现在考虑如何处理选择两朵云。
这里有两种情况。
第一种是两朵云无交,那么贡献就是\(len[i]+len[j]\)。
否则的话有交,并且交的部分只被这两朵云所覆盖,不能发现这样子的情况不会超过\(n\)个,这个东西可以拿出来暴力更新。
这样子我们可以预处理一个前缀的值,代表在这个位置之前选择云能够得到的最大空出来的时间。
对于第一种情况,可以按照价格大小构建一棵线段树,这样子每次就是区间查最大值。
对于第二种情况,找到这个区间然后把两个的贡献算在一起就好了。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<set>
#include<map>
using namespace std;
#define MAX 300300
inline int read()
{
	int x=0;bool t=false;char ch=getchar();
	while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
	if(ch=='-')t=true,ch=getchar();
	while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
	return t?-x:x;
}
struct Cloud{int l,r,c;}c[MAX];
bool operator<(Cloud a,Cloud b){return a.c<b.c;}
struct Node{int t,opt,id;}a[MAX<<1];
bool operator<(Node a,Node b){return a.t<b.t;}
struct Qry{int k,id;}q[MAX];
bool operator<(Qry a,Qry b){return a.k<b.k;}
int n,m,C,cnt,ans[MAX],len[MAX],cho[MAX];
namespace SegmentTree
{
#define lson (now<<1)
#define rson (now<<1|1)
	int t[MAX<<2];
	void Modify(int now,int l,int r,int p,int w)
	{
		if(l==r){t[now]=w;return;}
		int mid=(l+r)>>1;
		if(p<=mid)Modify(lson,l,mid,p,w);
		else Modify(rson,mid+1,r,p,w);
		t[now]=max(t[lson],t[rson]);
	}
	int Getmax(int now,int l,int r)
	{
		if(l==r)return l;
		int mid=(l+r)>>1;
		if(t[lson]>t[rson])return Getmax(lson,l,mid);
		else return Getmax(rson,mid+1,r);
	}
	int Query(int now,int l,int r,int L,int R)
	{
		if(L<=l&&r<=R)return Getmax(now,l,r);
		int mid=(l+r)>>1,ret=0;
		if(L<=mid)ret=Query(lson,l,mid,L,R);
		if(R>mid)
		{
			int x=Query(rson,mid+1,r,L,R);
			if(len[x]>len[ret])ret=x;
		}
		return ret;
	}
}
using namespace SegmentTree;
set<int> S;
map<int,int> M[MAX];
int Calc(int i,int j){if(i>j)swap(i,j);return len[i]+len[j]+M[i][j];}
int main()
{
	n=read();C=read();
	for(int i=1;i<=n;++i)c[i].l=read(),c[i].r=read(),c[i].c=read();
	sort(&c[1],&c[n+1]);c[n+1]=(Cloud){0,0,2000000000};
	for(int i=1;i<=n;++i)a[++cnt]=(Node){c[i].l,1,i},a[++cnt]=(Node){c[i].r,-1,i};
	sort(&a[1],&a[cnt+1]);a[++cnt]=(Node){2000000010,1,n+1};
	m=read();for(int i=1;i<=m;++i)q[i].k=read(),q[i].id=i;
	sort(&q[1],&q[m+1]);
	int Sum=0,Del=0;
	for(int i=1,tim=0,pos=1;i<=cnt;++i)
	{
		int l=a[i].t-tim;tim=a[i].t;
		if(!S.size())Sum+=l;
		else if(S.size()==1)
		{
			int x=*S.begin();
			len[x]+=l;cho[x]+=l;Modify(1,1,n,x,len[x]);
			if(c[x].c<=C)
			{
				int l=1,r=upper_bound(&c[1],&c[n+1],(Cloud){0,0,C-c[x].c})-c-1,val=len[x];
				if(l<=x&&x<=r)
				{
					if(l<=x-1)val=max(val,Calc(x,Query(1,1,n,l,x-1)));
					if(x+1<=r)val=max(val,Calc(x,Query(1,1,n,x+1,r)));
				}
				else if(l<=r)val=max(val,Calc(x,Query(1,1,n,l,r)));
				cho[x]=max(cho[x],val);
				Del=max(Del,cho[x]);
			}
		}
		else if(S.size()==2)
		{
			int x=*S.begin(),y=*S.rbegin();M[x][y]+=l;
			if(c[x].c+c[y].c<=C)
			{
				cho[x]=max(cho[x],Calc(x,y));
				cho[y]=max(cho[y],Calc(x,y));
				Del=max(Del,max(cho[x],cho[y]));
			}
		}
		while(pos<=m&&Del+Sum>=q[pos].k)ans[q[pos].id]=tim-(Del+Sum-q[pos].k),++pos;
		if(a[i].opt==1)S.insert(a[i].id);
		else S.erase(a[i].id);
	}
	for(int i=1;i<=m;++i)printf("%d\n",ans[i]);
	return 0;
}
【CF833E】Caramel Clouds(线段树)的更多相关文章
- CF833E Caramel Clouds
		题面 天上有$n$朵云,每朵云$i$会在时间$[l_i,r_i]$出现,你有$\text C$个糖果,你可以花费$c_i$个糖果让云$i$消失,同时需要保证你最多让两朵云消失.现在有$m$个独立的询问 ... 
- 【CF833E】Caramel Clouds
		[CF833E]Caramel Clouds 题面 洛谷 题目大意: 天上有\(n\)朵云,每朵云\(i\)会在时间\([li,ri]\)出现,你有\(C\)个糖果,你可以花费\(c_i\)个糖果让云 ... 
- bzoj3932--可持久化线段树
		题目大意: 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第 ... 
- codevs 1082 线段树练习 3(区间维护)
		codevs 1082 线段树练习 3 时间限制: 3 s 空间限制: 128000 KB 题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ... 
- codevs 1576 最长上升子序列的线段树优化
		题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ... 
- codevs 1080 线段树点修改
		先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ... 
- codevs 1082 线段树区间求和
		codevs 1082 线段树练习3 链接:http://codevs.cn/problem/1082/ sumv是维护求和的线段树,addv是标记这歌节点所在区间还需要加上的值. 我的线段树写法在运 ... 
- PYOJ 44. 【HNSDFZ2016 #6】可持久化线段树
		#44. [HNSDFZ2016 #6]可持久化线段树 统计 描述 提交 自定义测试 题目描述 现有一序列 AA.您需要写一棵可持久化线段树,以实现如下操作: A v p x:对于版本v的序列,给 A ... 
- CF719E(线段树+矩阵快速幂)
		题意:给你一个数列a,a[i]表示斐波那契数列的下标为a[i],求区间对应斐波那契数列数字的和,还要求能够维护对区间内所有下标加d的操作 分析:线段树 线段树的每个节点表示(f[i],f[i-1])这 ... 
随机推荐
- Dubbo负载均衡与集群容错机制
			1 Dubbo简介 Dubbo是一款高性能.轻量级的开源Java RPC框架,它提供了三大核心能力:面向接口的远程方法调用,智能容错和负载均衡,以及服务自动注册和发现. 作为一个轻量级RPC框架,D ... 
- WebPack牛刀小试
			现在页面的功能和需求越来越复杂,繁复杂乱的JavaScript代码和一大堆的依赖包都需要包含在前端页面中.如果还用手动处理就有点像在现代战场上使用小米加步枪的味道了. 为了减小开发的复杂度,前端社区涌 ... 
- Individual Project
			这次我自己完成了一个小小的项目,课可以把这篇随笔当做一次实验报告,主要的内容是用JUnit进行单元测试.由于我的技术太弱了,就在博客园里“求师”,按照大神的方法慢慢把这些东西写了下啦来. 不知道怎么搞 ... 
- 10-vue的介绍
			vue的作者叫尤雨溪,中国人.自认为很牛逼的人物,也是我的崇拜之神. 关于他本人的认知,希望大家读一下这篇关于他的文章,或许你会对语言,技术,产生浓厚的兴趣.https://mp.weixin.qq. ... 
- 【转帖】Linux定时任务Crontab命令详解
			Linux定时任务Crontab命令详解 https://www.cnblogs.com/intval/p/5763929.html 知道有crontab 以及 at 命令 改天仔细学习一下 讲sys ... 
- 阿里p3c(代码规范,eclipse插件、模版,idea插件)
			阿里p3c 一.说明 代码规范检查插件p3c,是根据<阿里巴巴Java开发手册>转化而成的自动化插件. (高级黑:P-3C“Orion”,反潜巡逻机,阿里大概取p3c先进,监测,发现潜在问 ... 
- Java Annotation详解 理解和使用Annotation
			系统中用到了java注解: 查了一下如何使用注解,到底注解是什么: (1)创建方法:MsgTrace Java Class==> 在Create New Class中: name:输入MsgTr ... 
- 初识Xml。
			/* * 一.Xml? * * 1.是什么? * Extensible markup Language 可拓展标记性语言 * 功能是 储存数据 * 1.配置文件 * 2.在网络中传输数据 * xml和 ... 
- python安装与配置
			首先下载python地址: https://www.python.org/downloads/release/python-361/ 下载页面中有多个版本: web-based installer 是 ... 
- 1064 - You have an error in your SQL syntax;
			mysql 1064 错误: SQL语法错误,check the manual that corresponds to your MySQL server version for the right ... 
