洛谷1937 仓配置 贪心+线段树

给出 n ≤ 1 e 5 n\leq1e5 n1e5个数,以及 m ≤ 1 e 5 m\leq1e5 m1e5个区间,然后每个区间都会占用一个位置,问最多能满足多少个区间。
将区间按照右端点排序,然后贪心判断,线段树区间减,维护区间最小值。
贪心的正确性:考虑两个区间 [ l 1 , r 1 ] [l_1,r_1] [l1,r1] [ l 2 , r 2 ] [l_2,r_2] [l2,r2],而且 r 1 < r 2 r_1<r_2 r1<r2。此时先选第一个区间更优。因为同样是多选择出来一个区间,而我选 [ l 1 , r 1 ] [l_1,r_1] [l1,r1]对后面区间的影响会更小。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int inf=0x3f3f3f3f;
const int N=1e5+7; 
struct Query {
	int l,r;
	bool operator <(const Query &rhs) const {
		return r<rhs.r;
	}
}q[N];
int a[N]; 
int mi[N<<2],sub[N<<2];
void pushup(int rt) {
	mi[rt]=min(mi[rt<<1],mi[rt<<1|1]);
}
void pushdown(int rt,int l,int mid,int r) {
	if(sub[rt]) {
		mi[rt<<1]-=sub[rt];
		mi[rt<<1|1]-=sub[rt];
		sub[rt<<1]+=sub[rt];
		sub[rt<<1|1]+=sub[rt];
		sub[rt]=0;
	}
}
void build(int rt,int l,int r) {
	if(l==r) {
		mi[rt]=a[l];
		return;
	}
	int mid=(l+r)>>1;
	build(rt<<1,l,mid);
	build(rt<<1|1,mid+1,r);
	pushup(rt);
} 
void modify(int rt,int l,int r,int L,int R) {
	if(R<l||L>r) return;
	if(L<=l&&r<=R) {
		sub[rt]++;
		mi[rt]--;
		return;
	}
	int mid=(l+r)>>1;
	pushdown(rt,l,mid,r);
	modify(rt<<1,l,mid,L,R);
	modify(rt<<1|1,mid+1,r,L,R);
	pushup(rt);
}
int query(int rt,int l,int r,int L,int R) {
	if(R<l||L>r) return 1e9;
	if(L<=l&&r<=R) return mi[rt];
	int mid=(l+r)>>1;
	pushdown(rt,l,mid,r);
	return min(query(rt<<1,l,mid,L,R),query(rt<<1|1,mid+1,r,L,R));
}
int main() {
	int n,m;
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;i++)
		scanf("%d",&a[i]);
	build(1,1,n);
	for(int i=1;i<=m;i++) 
		scanf("%d%d",&q[i].l,&q[i].r);
	sort(q+1,q+1+m);
	int ans=0;
	for(int i=1;i<=m;i++) {
		int v=query(1,1,n,q[i].l,q[i].r);
		if(v>=1) ++ans,modify(1,1,n,q[i].l,q[i].r);
	}
	printf("%d\n",ans);
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值