题目链接:http://www.spoj.com/problems/GSS5/

题意:给出一个数列。每次查询最大子段和Sum[i,j],其中i和j满足x1<=i<=y1,x2<=j<=y2,x1<=x2,y1<=y2。

思路:线段树的节点[L,R]保存LMax,RMax,Max,sum,表示左起最大值、右起最大值、区间最大值、区间数字和。更新比较简单。下面说查询。另外设置三个函数,可以查询任意区间[L,R]的最大值,以L开始向右最多到R的最大值、以R开始向左最多到L的最大值,设其分别为Q(L,R),QL(L,R),QR(L,R)。查询分两种情况讨论(设输入数组为d,S[i]表示d的前i项和):

(1)x2>y1: ans=QR(x1,y1)+S[x2-1]-S[y1]+QL(x2,y2);

(2)x2<=y1,那么有x1<=x2<=y1<=y2,则答案为下面三种情况的最大值:

a、x1<=i<=x2<=j<=y2,ans=QR(x1,x2)+QL(x2,y2)-d[x2];

b、x2<=i<=j<=y1,ans=Q(x2,y1);

c、x1<=i<=y1<=j<=y2,ans=QR(x1,y1)+QL(y1,y2)-d[y1]。

struct node
{
    int LMax,RMax,Max,sum;
    int L,R;
};

node a[N<<2];
int d[N],S[N];
int n;

void pushUp(int t)
{
    if(a[t].L==a[t].R) return;

a[t].LMax=max(a[t*2].LMax,a[t*2].sum+a[t*2+1].LMax);
    a[t].RMax=max(a[t*2+1].RMax,a[t*2+1].sum+a[t*2].RMax);
    a[t].sum=a[t*2].sum+a[t*2+1].sum;
    a[t].Max=max3(a[t*2].Max,a[t*2+1].Max,a[t*2].RMax+a[t*2+1].LMax);
}

void build(int t,int L,int R)
{
    a[t].L=L;
    a[t].R=R;
    if(L==R)
    {
        a[t].sum=d[L];
        a[t].LMax=a[t].RMax=a[t].Max=d[L];
        return;
    }
    int mid=(L+R)>>1;
    build(t*2,L,mid);
    build(t*2+1,mid+1,R);
    pushUp(t);
}

int queryL(int t,int L,int R)
{
    if(a[t].L==L&&a[t].R==R) return a[t].LMax;

int mid=(a[t].L+a[t].R)>>1;
    int ans;
    if(R<=mid) ans=queryL(t*2,L,R);
    else if(L>mid) ans=queryL(t*2+1,L,R);
    else
    {
        ans=max(queryL(t*2,L,mid),S[mid]-S[L-1]+queryL(t*2+1,mid+1,R));
    }
    return ans;
}

int queryR(int t,int L,int R)
{
    if(a[t].L==L&&a[t].R==R) return a[t].RMax;

int mid=(a[t].L+a[t].R)>>1;
    int ans;
    if(R<=mid) ans=queryR(t*2,L,R);
    else if(L>mid) ans=queryR(t*2+1,L,R);
    else
    {
        ans=max(queryR(t*2,L,mid)+S[R]-S[mid],queryR(t*2+1,mid+1,R));
    }
    return ans;
}

int query(int t,int L,int R)
{
    if(a[t].L==L&&a[t].R==R) return a[t].Max;

int mid=(a[t].L+a[t].R)>>1;
    int ans;
    if(R<=mid) ans=query(t*2,L,R);
    else if(L>mid) ans=query(t*2+1,L,R);
    else
    {
        int x=query(t*2,L,mid);
        int y=query(t*2+1,mid+1,R);
        ans=max(x,y);
        ans=max(ans,queryR(t*2,L,mid)+queryL(t*2+1,mid+1,R));
    }
    return ans;
}

int cal(int x1,int y1,int x2,int y2)
{
    int a,b,c;
    if(x2>y1)
    {
        a=queryR(1,x1,y1);
        b=queryL(1,x2,y2);
        return a+b+S[x2-1]-S[y1];
    }
    int ans;
    a=queryR(1,x1,x2);
    b=queryL(1,x2,y2);
    ans=a+b-d[x2];

a=query(1,x2,y1);
    ans=max(ans,a);

a=queryR(1,x1,y1);
    b=queryL(1,y1,y2);
    ans=max(ans,a+b-d[y1]);

return ans;
}

int main()
{
    rush()
    {
        RD(n);
        int i;
        FOR1(i,n) RD(d[i]),S[i]=S[i-1]+d[i];
        build(1,1,n);
        int x1,y1,x2,y2;
        int m;
        RD(m);
        while(m--)
        {
            RD(x1,y1); RD(x2,y2);
            PR(cal(x1,y1,x2,y2));
        }
    }
}

SPOJ 2916 Can you answer these queries V(线段树-分类讨论)的更多相关文章

  1. GSS5 spoj 2916. Can you answer these queries V 线段树

    gss5 Can you answer these queries V 给出数列a1...an,询问时给出: Query(x1,y1,x2,y2) = Max { A[i]+A[i+1]+...+A[ ...

  2. SPOJ GSS5 Can you answer these queries V ——线段树

    [题目分析] GSS1上增加区间左右端点的限制. 直接分类讨论就好了. [代码] #include <cstdio> #include <cstring> #include & ...

  3. SPOJ GSS3 Can you answer these queries III[线段树]

    SPOJ - GSS3 Can you answer these queries III Description You are given a sequence A of N (N <= 50 ...

  4. SPOJ 1557. Can you answer these queries II 线段树

    Can you answer these queries II Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 https://www.spoj.com/pr ...

  5. bzoj 2482: [Spoj GSS2] Can you answer these queries II 线段树

    2482: [Spoj1557] Can you answer these queries II Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 145 ...

  6. SPOJ GSS1 - Can you answer these queries I(线段树维护GSS)

    Can you answer these queries I SPOJ - GSS1 You are given a sequence A[1], A[2], -, A[N] . ( |A[i]| ≤ ...

  7. SPOJ GSS1 Can you answer these queries I[线段树]

    Description You are given a sequence A[1], A[2], ..., A[N] . ( |A[i]| ≤ 15007 , 1 ≤ N ≤ 50000 ). A q ...

  8. 【SP2916】Can you answer these queries V - 线段树

    题面 You are given a sequence \(a_1,a_2,...,a_n\). (\(|A[i]| \leq 10000 , 1 \leq N \leq 10000\)). A qu ...

  9. SPOJ GSS3 Can you answer these queries III ——线段树

    [题目分析] GSS1的基础上增加修改操作. 同理线段树即可,多写一个函数就好了. [代码] #include <cstdio> #include <cstring> #inc ...

随机推荐

  1. 2013ACM/ICPC亚洲区南京站现场赛——题目重现

    GPA http://acm.hdu.edu.cn/showproblem.php?pid=4802 签到题,输入两个表,注意细心点就行了. #include<cstdio> #inclu ...

  2. ActiveMQ主从配置

    这种方式有个问题,activemq1有消息没消费完但是突然宕机,虽然程序会自动连到activemq2.但是activemq1的消息只有等机器恢复后才会被消费. 1.启动:我这里使用的是apache-a ...

  3. frequentism-and-bayesianism-chs-ii

    frequentism-and-bayesianism-chs-ii 频率主义 vs 贝叶斯主义 II:当结果不同时   这个notebook出自Pythonic Perambulations的博文  ...

  4. sigaction 函数

    本文主要参考<unix环境高级编程>   sigaction函数的功能是检查或修改与指定信号相关联的处理动作(可同时两种操作).   int sigaction(int signo,con ...

  5. 关于JS APP

    多屏screen, JS如何路由,如何换页,导航.通过JS来实现. 当前页面的逻辑通过JS来实现.HTML DOM, Event, Widget. 核心在于function. JS 不仅仅是DOM, ...

  6. CSS 外边距(margin)重叠及防止方法

    边界重叠是指两个或多个盒子(可能相邻也可能嵌套)的相邻边界(其间没有任何非空内容.补白.边框)重合在一起而形成一个单一边界. 两个或多个块级盒子的垂直相邻边界会重合.结果的边界宽度是相邻边界宽度中最大 ...

  7. ECMAScript 5.1中对属性的操作

    此操作只针对新版本浏览器. Object.defineProperty(obj,'name',{configurable:true/false,enumberable:true/false,value ...

  8. C内联汇编

    用C写程序比直接用汇编写程序更简洁,可读性更好,但效率可能不如汇编程序,因为C程序毕竟要经由编译器生成汇编代码,尽管现代编译器的优化已经做得很好了,但还是不如手写的汇编代码.另外,有些平台相关的指令必 ...

  9. SQL技术内幕-8 使用WITH AS提高性能简化嵌套SQL

    --本文来源:http://www.cnblogs.com/fygh/archive/2011/08/31/2160266.html 一.WITH AS的含义 WITH AS短语,也叫做子查询部分(s ...

  10. Struts2 直接返回字符串(可用于json)

    struts2可以在method中直接返回一个字符串而不是视图.让Action中的返回字符串的方法,直接return null;并在return之前用输出流输出字符串就可以了,跟servlet输出HT ...