poj-2750 Potted Flower

88 篇文章 0 订阅
30 篇文章 0 订阅

题意:

给出一个n个结点的环,在这个环上取一段区间(不能是整个环),使权值和最大;

m次修改某个结点权值,每次修改后输出最大值;

n,m<=100000;


题解:

这道题的难点主要在于给出的是一个环而不是序列;

所以选择一个点把环拆成序列,那么就可以把问题转化成:

在一个序列中,取一段子序列,或两边都取两段序列,是权值最大;

这个答案就是在这个序列里取区间的max(最大值ma,序列和sum-最小值mi);

这些都可以用线段树维护;

但是,这道题不允许取整段序列,所以直接这么搞是不行的;

考虑什么时候会取整段序列呢,那也就是没有负数的时候(有负数就直接不取它就有了更优解);

没有负数也就意味着sum==ma,于是特判就可以了;


代码:


#include<stdio.h>
#include<string.h>
#include<algorithm>
#define N 110000
#define lson l,mid,no<<1
#define rson mid+1,r,no<<1|1
using namespace std;
int sum[N<<2],ma[N<<2],mi[N<<2],miL[N<<2],miR[N<<2],maL[N<<2],maR[N<<2];
void Pushup(int no)
{
	sum[no]=sum[no<<1]+sum[no<<1|1];
	maL[no]=max(maL[no<<1],sum[no<<1]+maL[no<<1|1]);
	miL[no]=min(miL[no<<1],sum[no<<1]+miL[no<<1|1]);
	maR[no]=max(maR[no<<1|1],sum[no<<1|1]+maR[no<<1]);
	miR[no]=min(miR[no<<1|1],sum[no<<1|1]+miR[no<<1]);
	ma[no]=max(max(ma[no<<1],ma[no<<1|1]),maR[no<<1]+maL[no<<1|1]);
	mi[no]=min(min(mi[no<<1],mi[no<<1|1]),miR[no<<1]+miL[no<<1|1]);
}
void build(int l,int r,int no)
{
	if(l==r)
	{
		scanf("%d",&sum[no]);
		ma[no]=mi[no]=miL[no]=miR[no]=maL[no]=maR[no]=sum[no];
	}
	else
	{
		int mid=(l+r)>>1;
		build(lson);
		build(rson);
		Pushup(no);
	}
}
void update(int l,int r,int no,int k,int val)
{
	if(l==r)
	{
		sum[no]=ma[no]=mi[no]=miL[no]=miR[no]=maL[no]=maR[no]=val;
	}
	else
	{
		int mid=(l+r)>>1;
		if(k<=mid)	update(lson,k,val);
		else		update(rson,k,val);
		Pushup(no);
	}
}
int main()
{
	int n,m,i,j,k,x,y,l,r,ans;
	scanf("%d",&n);
	build(1,n,1);
	scanf("%d",&m);
	for(i=1;i<=m;i++)
	{
		scanf("%d%d",&x,&y);
		update(1,n,1,x,y);
		if(sum[1]==ma[1])
			printf("%d\n",sum[1]-mi[1]);
		else
			printf("%d\n",max(ma[1],sum[1]-mi[1]));
	}
	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值