首页 > 其他 > 详细

4540: [Hnoi2016]序列|莫队+ST表

时间:2016-04-29 20:03:33      阅读:319      评论:0      收藏:0      [点我收藏+]

考虑现在已经知道了[l,r]的答案新添入一个r+1如何更新答案
也就是右端点在r+1处左端点在l..r+1之间的所有的子序列的答案
可以找出l..r中最小的数的位置p,然后p以及p左侧作为左端点的答案就可以直接计算了
考虑左端点在p+1....r+1时对答案的贡献,可以与处理一个前缀和Si表示以i为右端点的所有子序列的答案之和
那么左端点在p+1....r+1时对答案的贡献就是Sr+1?Sp
其他端点移动的做法也同理
为什么我的莫队跑了17s,而网上的其他莫队只需要5s,人傻自带三倍常数QWQ

#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<vector>
#include<cmath>
#include<queue>
#include<ctime>
#include<set>
#include<map>
#define N 110000
#define ll long long
using namespace std;
int sc()
{
    int i=0,f=1; char c=getchar();
    while(c>‘9‘||c<‘0‘){if(c==‘-‘)f=-1;c=getchar();}
    while(c>=‘0‘&&c<=‘9‘)i=i*10+c-‘0‘,c=getchar();
    return i*f;
}
struct W{int l,r,p;} b[N];
long long sl[N],sr[N],ans[N],now;
int bl[N],l[N],r[N],a[N],st[N][17],q[N];
int n,Q,block;
void pre()
{
    for(int i=1;i<=n;i++) st[i][0]=i;
    for(int k=1;(1<<k)<=n;k++)
        for(int i=1;i<=n;i++)
            if(i+(1<<k)>n+1)break;
            else if(a[st[i][k-1]]<a[st[i+(1<<k-1)][k-1]]) st[i][k]=st[i][k-1];
            else st[i][k]=st[i+(1<<k-1)][k-1];
    int qr=0;
    for(int i=1;i<=n;i++)
    {
        while(qr&&a[q[qr]]>=a[i])qr--;
        l[i]=q[qr]; q[++qr]=i;
        sl[i]=sl[l[i]]+1ll*a[i]*(i-l[i]);
    }
    q[qr=0]=n+1;
    for(int i=n;i>=1;i--)
    {
        while(qr&&a[q[qr]]>a[i])qr--;
        r[i]=q[qr]; q[++qr]=i;
        sr[i]=sr[r[i]]+1ll*a[i]*(r[i]-i);
    }
}
bool cmp(W a,W b)
{
    return bl[a.l]<bl[b.l]||(bl[a.l]==bl[b.l]&&a.r<b.r);
}
int get_min(int l,int r)
{
    int k=log2(r-l+1);
    int L=st[l][k],R=st[r-(1<<k)+1][k];
    return a[L]<a[R]?L:R;
}
long long call(int l,int r)
{
    int p=get_min(l,r);
    return 1ll*(r-p+1)*a[p]+sr[l]-sr[p];
}
long long calr(int l,int r)
{
    int p=get_min(l,r);
    return 1ll*(p-l+1)*a[p]+sl[r]-sl[p];
}
int main()
{
    n=sc(),Q=sc();block=sqrt(n);
    for(int i=1;i<=n;i++)
        a[i]=sc(),bl[i]=(i-1)/block;
    pre();
    for(int i=1;i<=Q;i++)
        b[i].l=sc(),b[i].r=sc(),b[i].p=i;
    sort(b+1,b+Q+1,cmp);
    int L=1,R=0;
    for(int i=1;i<=Q;i++)
    {
        while(L>b[i].l)now+=call(L-1,R),L--;
        while(R<b[i].r)now+=calr(L,R+1),R++;
        while(L<b[i].l)now-=call(L,R),L++;
        while(R>b[i].r)now-=calr(L,R),R--;
        ans[b[i].p]=now;
    }
    for(int i=1;i<=Q;i++)
        printf("%lld\n",ans[i]);
    return 0;
}

4540: [Hnoi2016]序列|莫队+ST表

原文:http://blog.csdn.net/ws_yzy/article/details/51224653

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!