LA3938 "Ray, Pass me the dishes!" (线段树区间合并)

题目:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=22105

题意:给定整数n和m,给出一个n个元素的序列,查询m次给定区间[L,R]的最大连续和的位置[x,y],有多个区间输出x最小的,还有多个的话输出y最小的。

分析:每个节点存8个信息,最大连续和、最大后缀和、最大前缀和、区间和、前缀末位置、后缀首位置、最大连续和的首位置和末位置。

最大连续和=max(lson最大连续和,rson最大连续和,lson最大后缀+rson最大前缀)

最大前缀和=max(lson最大前缀,lson区间和+rson最大前缀和)

最大后缀和=max(rson最大后缀,lson最大后缀+rson区间和)

查询的时候将节点合并就行了。

代码:

#include <iostream>
#include <cstdio>
using namespace std;
const int maxn = 5e5+6;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
struct node
{
	long long Sum,MaxSum;
	long long Suf,Pre;
	int pos_p,pos_s,s_l,s_r;
}ans;
struct segtree
{
	node tree[maxn<<2];
	node U(node &a,node &b)
	{
		node ret;
		if(a.MaxSum>=b.MaxSum) //以下修改最大连续和,及标记位置 
		{
			ret.MaxSum=a.MaxSum;
			ret.s_l=a.s_l;
			ret.s_r=a.s_r;
		}
		else
		{
			ret.MaxSum=b.MaxSum;
			ret.s_l=b.s_l;
			ret.s_r=b.s_r;
		}
		if(ret.MaxSum<a.Suf+b.Pre || (ret.MaxSum==a.Suf+b.Pre && a.pos_s<ret.s_l) || 
		   (ret.MaxSum==a.Suf+b.Pre && ret.s_l==a.pos_s && ret.s_r>b.pos_p))
		{
			ret.MaxSum=a.Suf+b.Pre;
			ret.s_l=a.pos_s;
			ret.s_r=b.pos_p;
		} 
		if(a.Pre>=a.Sum+b.Pre) //以下修改最大前缀,及标记位置 
		{
			ret.Pre=a.Pre;
			ret.pos_p=a.pos_p;
		}
		else
		{
			ret.Pre=a.Sum+b.Pre;
			ret.pos_p=b.pos_p;
		}
		if(b.Suf>a.Suf+b.Sum) //以下修改最大后缀,及标记位置 
		{
			ret.Suf=b.Suf;
			ret.pos_s=b.pos_s;
		}
		else
		{
			ret.Suf=a.Suf+b.Sum;
			ret.pos_s=a.pos_s;
		}
		ret.Sum=a.Sum+b.Sum;  //修改区间和 
		return ret;
	}
	void build(int l,int r,int rt)
	{
		if(l==r)
		{
			scanf("%lld",&tree[rt].Pre);  //坑啊C++11不能I64d 
			tree[rt].MaxSum=tree[rt].Sum=tree[rt].Suf=tree[rt].Pre;
			tree[rt].pos_p=tree[rt].pos_s=tree[rt].s_l=tree[rt].s_r=r;
			return ;
		}
		int m=(l+r)>>1;
		build(lson);
		build(rson);
		tree[rt]=U(tree[rt<<1],tree[rt<<1|1]);
	}
	node query(int L,int R,int l,int r,int rt)
	{
		if(L<=l && r<=R)
			return tree[rt];
		int m=(l+r)>>1,fg1(0),fg2(0);
		node ans1,ans2;
		if(L<=m)
			ans1=query(L,R,lson),fg1=1;
		if(R>m)
			ans2=query(L,R,rson),fg2=1;
		if(!fg1)
			return ans2;
		if(!fg2)
			return ans1;
		return U(ans1,ans2);
	}
}T;
int main()
{
	int n,m,x,y,ncase=1;
	while(scanf("%d%d",&n,&m)!=EOF)
	{
		T.build(1,n,1);
		printf("Case %d:\n",ncase++);
		while(m--)
		{
			scanf("%d%d",&x,&y);
			ans=T.query(x,y,1,n,1);
			printf("%d %d\n",ans.s_l,ans.s_r);
		}
	}
	return 0;
}

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值