纪中游记(九)

20190813

写在前面的话

哇~前两题是真的简单,后两题是真的难

T0&T1

初&复&终

桶排模板+hash模板

没什么好说的(虽然第一题由于粗心只拿了十分)

T2

单点修改当然是树状数组哒,可数据范围让O(n)的单调队列也望尘莫及

没办法,我又不会线段树,只好硬着头皮上O(n*m)

只得了10分,意料之中

果不其然,这题正解真是用线段树维护区间最大子段和,

(菜鸡HHHG看着1800多的线段树模板代码瑟瑟发抖)

不可能的,太难了,这辈子都不可能码出来的(已疯)

2019.08.14.14.50

我完成了(吼吼吼)(感谢MHY大佬的讲解)

果然昨天的线段树模板没有白打

值此兴奋之际,我来讲一下这道题

------------华丽的分割线------------

我们先看题,单点修改再加上查询区间最大子段和

前面的伏笔都指向了一个神奇的数据结构——线段树

而线段树需要维护什么呢?你首先想到的一定是区间最大子段和,没错

那又如何合并呢?看图:

这样就很清晰了,但是题目中是环,所以我们还要维护一个区间最小子段和,最后答案就是:

ans=max(区间总和-区间最小子段和,区间最大子段和)

AC(困)(难)(重)(重)

 

  1 #include<bits/stdc++.h>
  2 #define lll long long
  3 using namespace std;
  4 int n,m;
  5 struct oier
  6 {
  7     lll xl,xr,xm,nl,nr,nm,sum;//最大前缀和,最大后缀和,最大子段和,最小前缀和,最小后缀和,最小子段和,区间和;
  8 }lt[400005]; 
  9 lll a[100005];
 10 inline int ls(int x)
 11 {
 12     return x<<1;
 13 }
 14 inline int rs(int x)
 15 {
 16     return x<<1|1;
 17 }
 18 void pu(int x)
 19 {
 20     lt[x].sum=lt[ls(x)].sum+lt[rs(x)].sum;
 21     lt[x].xl=max(lt[ls(x)].xl,lt[ls(x)].sum+lt[rs(x)].xl);
 22     lt[x].xr=max(lt[rs(x)].xr,lt[rs(x)].sum+lt[ls(x)].xr);
 23     lt[x].xm=max(lt[ls(x)].xr+lt[rs(x)].xl,max(lt[ls(x)].xm,lt[rs(x)].xm));
 24     lt[x].nl=min(lt[ls(x)].nl,lt[ls(x)].sum+lt[rs(x)].nl);
 25     lt[x].nr=min(lt[rs(x)].nr,lt[rs(x)].sum+lt[ls(x)].nr);
 26     lt[x].nm=min(lt[ls(x)].nr+lt[rs(x)].nl,min(lt[ls(x)].nm,lt[rs(x)].nm));
 27 }
 28 void build(lll x,lll l,lll r)
 29 {
 30     if(l==r)
 31     {
 32         lt[x].sum=lt[x].nl=lt[x].nm=lt[x].nr=lt[x].xl=lt[x].xm=lt[x].xr=a[l];
 33         return ;
 34     }
 35     lll mid=(l+r)>>1;
 36     build(ls(x),l,mid);
 37     build(rs(x),mid+1,r);
 38     pu(x);
 39 }
 40 inline void update(lll l,lll r,lll x,lll nn,lll k)
 41 {
 42     if(l==r&&l==nn)
 43     {
 44         lt[x].sum=lt[x].nl=lt[x].nm=lt[x].nr=lt[x].xl=lt[x].xm=lt[x].xr=k;
 45         return ;
 46     }
 47     lll mid=(l+r)>>1;
 48     if(nn<=mid) update(l,mid,ls(x),nn,k);
 49     if(nn>mid) update(mid+1,r,rs(x),nn,k);
 50     pu(x);
 51 }
 52 //oier query(lll xx,lll yy,lll l,lll r,lll x)
 53 //{
 54 //    oier num={-0x7fffffff,-0x7fffffff,-0x7fffffff,0,0x7fffffff,0x7fffffff,0x7fffffff} ,a={-0x7fffffff,-0x7fffffff,-0x7fffffff,0,0x7fffffff,0x7fffffff,0x7fffffff},b={-0x7fffffff,-0x7fffffff,-0x7fffffff,0,0x7fffffff,0x7fffffff,0x7fffffff};
 55 //    //if(l>=xx&&r<=yy) return lt[x];
 56 //    lll mid=(l+r)>>1;
 57 //    if(xx<=mid) a=query(xx,yy,l,mid,ls(x));
 58 //    if(yy>mid) b=query(xx,yy,mid+1,r,rs(x));
 59 //    num.sum=a.sum+b.sum;
 60 //    num.xl=max(a.xl,a.sum+b.xl);
 61 //    num.xr=max(b.xr,b.sum+a.xr);
 62 //    num.xm=max(a.xr+b.xl,max(a.xm,b.xm));
 63 //    num.nl=min(a.nl,a.sum+b.nl);
 64 //    num.nr=min(b.nr,b.sum+a.nr);
 65 //    num.nm=min(a.nr+b.nl,min(a.nm,b.nm));
 66 //    return num;
 67 //}//无用的函数 
 68 inline lll read()
 69 {
 70     lll x=0,f=1;
 71     char ch=getchar();
 72     while(ch<'0'||ch>'9')
 73     {
 74         if(ch=='-')
 75             f=-1;
 76         ch=getchar();
 77     }
 78     while(ch>='0'&&ch<='9')
 79     {
 80         x=(x<<1)+(x<<3)+ch-'0';
 81         ch=getchar();
 82     }
 83     return x*f;
 84 }
 85 int main()
 86 {
 87     n=read();
 88     for(int i=1;i<=n;i++)
 89     {
 90         a[i]=read();
 91     }
 92     build(1,1,n);
 93     m=read();
 94     while(m--)
 95     {
 96             lll xx,zz;
 97             xx=read(),zz=read();
 98             update(1,n,1,xx,zz);
 99             //oier ui=query(1,n,1,n,1);
100             printf("%lld\n",max(lt[1].sum-lt[1].nm,lt[1].xm));
101     }
102     return 0;
103 }
AChhh

 

 

T3

直接暴力,成功骗得40分

 

posted @ 2019-08-13 21:52  HHHG  阅读(193)  评论(0编辑  收藏  举报