bzoj5055: 膜法师

来源:互联网 发布:网络推广客服好做吗 编辑:程序博客网 时间:2024/04/27 23:54

Description

在经历过1e9次大型战争后的宇宙中现在还剩下n个完美维度,
现在来自多元宇宙的膜法师,想偷取其中的三个维度为伟大的长者续秒,
显然,他能为长者所续的时间,为这三个维度上能量的乘积,
但目前的宇宙很不乐观,胡乱偷取可能造成维度的崩溃,
所以,他必须按逆序偷取这些维度,且在偷取中,
每次偷取的维度的能量必须严格小于他上次偷取的能量,
由于膜法师生活在多元宇宙,所以他可以让所有可能的偷取方案全部发生
题目描述
但他数学不好,所以找到了你帮他求出能为长者续几秒,
你要做的,就是在给定的维度序列a中,
求出所有满足i

题解

其实就是找一个数前面比他小的和,后面比他大的和。。
然后离散化一下,之后树状数组,线段树什么随便用用就好了
然而我比较懒。。
不想离散化。。
于是偷偷地用线段树动态开点。。
但是空间过不去。。
于是偷偷地自己开小一点空间。。
数据没卡,就A了TAT

#include<cstdio>#include<cstring>typedef long long LL;typedef unsigned int Int;const int MOD=19260817;Int mymax (Int x,Int y){return x>y?x:y;}const int N=300005;Int maxx=0;int n;Int a[N];LL b[N];LL c[N*20];int s1[N*20],s2[N*20];LL get (int now,Int l,Int r,Int L,Int R){    if (now==0) return 0;    if (l==L&&r==R)        return c[now];    Int mid=(l+r)>>1;    if (R<=mid) return get(s1[now],l,mid,L,R);    else if (L>mid) return get(s2[now],mid+1,r,L,R);    else return get(s1[now],l,mid,L,mid)+get(s2[now],mid+1,r,mid+1,R);}int num=1;void change (int &now,Int l,Int r,Int x){    if (now==0) now=++num;    c[now]=(c[now]+x)%MOD;    if (l==r) return ;    Int mid=(l+r)>>1;    if (x<=mid) change(s1[now],l,mid,x);    else change(s2[now],mid+1,r,x);}int main(){    scanf("%d",&n);    for (int u=1;u<=n;u++)    {        scanf("%u",&a[u]);        maxx=mymax(maxx,a[u]);    }    int rt=1;    for (int u=1;u<=n;u++)    {        if (a[u]>0) b[u]=a[u]*get(1,1,maxx,1,a[u]-1)%MOD;        change(rt,1,maxx,a[u]);    }    num=1;memset(c,0,sizeof(c));    memset(s1,0,sizeof(s1));    memset(s2,0,sizeof(s2));    LL ans=0;    for (int u=n;u>=1;u--)    {        if (a[u]<maxx) ans=(ans+(b[u]*get(1,1,maxx,a[u]+1,maxx))%MOD)%MOD;        change(rt,1,maxx,a[u]);    }    printf("%lld\n",ans);    return 0;}
原创粉丝点击