Norma

传送门

题意简化:

定义一个区间的贡献为 \(max*min*len\),求给定序列中所有子区间的总贡献和

题解

考虑 \(O(n*log_2n)\) 的复杂度的做法

数据结构??? yzhx太菜了,不会怎么O(n)枚举所有区间

还是考虑分治吧

每次对于每个区间的贡献则等于:

左半边区间的贡献+右半边区间的贡献+跨越mid的区间贡献

所以现在分治的大体思路就出来了,先递归处理左右半边,在加上跨越左右区间的影响

我们再来观察这个式子: \(max*min*len\)

想想有什么可优化的部分呢?

显然,对于每个区间的max和min在适当扩张区间长度的情况下是不会改变的

所以我们暴力求的话则会重复求很多次

有了大致思路,那么,我们现在来考虑怎么处理跨mid的子区间贡献

设当前区间的左端点为 L ,右端点为 R , mid 为该区间的中间位置

对于跨越mid的子区间

枚举子区间左端点 i ,先假设它的右端点就是mid,那么我们再慢慢向右拓展右端点,并加入答案

设: 子区间i~j的最大值为max,最小值为min,

在拓展过程中 min 第一次改变的位置是 p , max 第一次改变的位置是 q , p<q (反过来也差不多)

所以所有以 i 为左端点,右端点j在 mid+1 ~ R 之间的子区间都可以被分为三种类型:

j < p 时 ( \(min*max\) 不变) :

\[ans+= min*max* \sum_{j=mid+1}^{p-1}(j-i+1)
\]

p<=j<q 时(只有 min 和区间长度发生改变):

\[ans+= max * \sum_{j=p}^{q-1} min[j] * (j-i+1)
\]

--> $$ ans+=max* \sum_{j=p}^{q-1} min[j] j+max * (1-i)\sum_{j=p}^{q-1} min[j]) $$

q<j<=r 时(全都改变):

\[ans+= \sum_{j=p}^{q-1} max[j]*min[j] * (j-i+1)
\]

--> $$ ans+= \sum_{j=p}^{q-1} max[j]min[j] j+(1-i)\sum_{j=p}^{q-1} max[j]min[j]) $$

然后我们再把 $$max[i],min[i],max[i]i,min[i]i,max[i]min[i],max[i]min[i]*i $$都用前缀和记录就好

代码

#include<bits/stdc++.h>
using namespace std;
#define re register
#define in inline
#define get getchar()
#define ll long long
in int read()
{
int t=0; char ch=get;
while(ch<'0' || ch>'9') ch=get;
while(ch<='9' && ch>='0') t=t*10+ch-'0', ch=get;
return t;
}
const int mod=1e9;
const int _=500002;
int n;
ll mul_mx[_],mul_mn[_],ans,sum_mx[_],sum_mn[_],mx[_],mn[_],a[_],mnmx[_],mul_mnmx[_];
in ll add(ll a,ll b) { //加法取模
return (((a%mod)+(b%mod))+mod)%mod;
}
in ll mul(ll a,ll b) { //乘法取模
return ((a%mod*b%mod)+mod)%mod;
}
in ll getsum(ll a,ll b) { //高斯求和
return ((a+b)*(b-a+1)/2)%mod;
}
in void work(int l,int r)
{
// cout<<l<<' '<<r<<" :: "<<endl;
if(l==r) { ans=add(ans,mul(a[l],a[l]));return;}
int mid=l+r>>1;
work(l,mid);
work(mid+1,r);
mx[mid]=mn[mid]=a[mid];
sum_mx[mid]=sum_mn[mid]=mul_mx[mid]=mul_mn[mid]=mul_mnmx[mid]=mnmx[mid]=0;
for(re int i=mid+1;i<=r;i++) {
mx[i]=max(mx[i-1],a[i]);
mn[i]=min(mn[i-1],a[i]);
sum_mx[i]=add(sum_mx[i-1],mx[i]);
sum_mn[i]=add(sum_mn[i-1],mn[i]);
mul_mx[i]=add(mul_mx[i-1],mul(mx[i],i));
mul_mn[i]=add(mul_mn[i-1],mul(mn[i],i));
mul_mnmx[i]=add(mul_mnmx[i-1],mul(mul(mx[i],mn[i]),i));
mnmx[i]=add(mnmx[i-1],mul(mx[i],mn[i]));
} //预处理
ll maxx=0,minn=0x3f3f3f3f3f3f3f3f;
for(re int p=mid+1,q=mid+1,i=mid;i>=l;i--)
{
minn=min(minn,a[i]),maxx=max(maxx,a[i]);
while(p<=r&&minn<a[p]) p++;
while(q<=r&&maxx>a[q]) q++; //找到p和q
if(p<q)
{
ans=add(ans,mul(mul(minn,maxx),getsum(mid-i+2,p-i)));
ans=add(ans,add(mul(maxx,add(mul_mn[q-1],-mul_mn[p-1])),-mul(mul(i-1,maxx),add(sum_mn[q-1],-sum_mn[p-1]))));
ans=add(ans,add(add(mul_mnmx[r],-mul_mnmx[q-1]),mul(add(1,-i),add(mnmx[r],-mnmx[q-1]))));
} //套上之前讲的三个式子
else
{
ans=add(ans,mul(mul(minn,maxx),getsum(mid-i+2,q-i)));
ans=add(ans,add(mul(minn,add(mul_mx[p-1],-mul_mx[q-1])),-mul(mul(i-1,minn),add(sum_mx[p-1],-sum_mx[q-1]))));
ans=add(ans,add(add(mul_mnmx[r],-mul_mnmx[p-1]),mul(add(1,-i),add(mnmx[r],-mnmx[p-1]))));
}
}
}
int main()
{
n=read();
for(re int i=1;i<=n;i++) a[i]=read();
work(1,n);
cout<<ans<<endl;
}

SP22343 Norma--序列分治的更多相关文章

  1. bzoj 3745 [Coci2015]Norma——序列分治

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3745 如果分治,就能在本层仅算过 mid 的区间了. 可以从中间到左边地遍历左边,给右边两个 ...

  2. [CF888E] Maximum Subsequence 序列分治

    早期作品,不喜轻喷. LG传送门 序列分治板子题. 切这道题用了好长时间,所以想发篇题解作为纪念 . 首先,我们认真观察题目数据(面向数据做题是个好习惯),发现题目的\(n\)竟然只有\(35\),我 ...

  3. 算法复习——序列分治(ssoj光荣的梦想)

    题目: 题目描述 Prince对他在这片大陆上维护的秩序感到满意,于是决定启程离开艾泽拉斯.在他动身之前,Prince决定赋予King_Bette最强大的能量以守护世界.保卫这里的平衡与和谐.在那个时 ...

  4. 洛谷SP22343 NORMA2 - Norma(分治,前缀和)

    洛谷题目传送门 这题推式子恶心..... 考虑分治,每次统计跨过\(mid\)的所有区间的答案和.\(i\)从\(mid-1\)到\(l\)枚举,统计以\(i\)为左端点的所有区间. 我们先维护好\( ...

  5. 【BZOJ3745】[Coci2015]Norma cdq分治

    [BZOJ3745][Coci2015]Norma Description Input 第1行,一个整数N: 第2~n+1行,每行一个整数表示序列a. Output 输出答案对10^9取模后的结果. ...

  6. NORMA2 - Norma [cdq分治]

    题面 洛谷 你有一个长度为n的序列,定义这个序列中每个区间的价值是 \(Cost(i,j)=Min(Ai...Aj)∗Max(Ai...Aj)∗(j−i+1)Cost(i,j)=Min(A_{i}.. ...

  7. BZOJ 3745: [Coci2015]Norma(分治)

    题意 给定一个正整数序列 \(a_1, a_2, \cdots, a_n\) ,求 \[ \sum_{i=1}^{n} \sum_{j=i}^{n} (j - i + 1) \min(a_i,a_{i ...

  8. UVA 11990 ``Dynamic'' Inversion (序列分治)

    26天以前做过的一道题,之前的做法是分治预处理,树套树在线修改,复杂度为O(nlogn+m*logn*logn),代码量较大. 本来想学习一下cdq分治的,看到论文上的凸包.斜率就暂时放一边了,只知道 ...

  9. bzoj 3745: [Coci2015]Norma【分治】

    参考:https://blog.csdn.net/lych_cys/article/details/51203960 真的不擅长这种-- 分治,对于一个(l,r),先递归求出(l,mid),(mid+ ...

随机推荐

  1. Go 数组&切片

    数组相关 在Go语言中,数组是一种容器相关的数据类型,用于存放多种相同类型的数据. 数组定义 在定义数组时,必须定义数组的类型以及长度,数组一经定义不可进行改变. 同时,数组的长度是按照元素个数进行统 ...

  2. Ubuntu部署和体验Nexus3

    关于Nexus 如下图,在局域网部署了Nexus之后,可以缓存中央仓库的jar,开发者开发的二方库发布到Nexus上,局域网内的其他人也可以从Nexus下载这些二方库使用: 环境信息 本次实战是在Li ...

  3. Windows控件的属性与事件

    Treeview控件重要属性和事件 属性 说明 Nodes Treeview控件中所有树节点 SelectdNode 当前Treeview控件中选定的树节点,如果当前没有选定树节点,返回值为null ...

  4. IDEA2020版最佳优化思路(中文界面)

    IDEA优化 基于当前最新版idea 2020.1版本进行设置 设置中文 在idea 2020.1版本后官方是支持中文啦 先上效果图 设置方法 这里需要下载官方的中文包 鼠标悬停提示 效果图 设置方法 ...

  5. 01 AS 首次编译执行项目过程中遇到的几个常见问题

    问题01 as打开时出现The environment variable JAVA_HOME (with The value of C:\Java\jdk1.8.0_101\bin) does not ...

  6. 【题解】CF1324F

    Question 题目大意:每个点不是黑点就是白点,求以每一个点为根时,选择出一个联通块,使得白点数与黑点数之差最大(白减黑). \(Solution\) 考虑先跑一遍\(dp\). 可以写出一个比较 ...

  7. pycharm 解决PEP8问题,配置autopep8到菜单栏

    autopep8是一个可以将Python代码自动排版为PEP8风格第三方包,使用它可以轻松地排版出格式优美整齐的代码.网络上有很多介绍如何在pycharm中配置autopep8的方案,但很多方案中还是 ...

  8. IDEA2020.2的破解

    第一种方式:http://code.39sd.cn/ 直接获取二维码: 第二种:下载破解工具(本方法只是提供个人学习使用) 1.下载2020.2的idea 链接:https://pan.baidu.c ...

  9. SCOI 2008 【奖励关】

    早上的考试一道都做不出,被教做人,心态爆炸ing...... 题目描述: 你正在玩你最喜欢的电子游戏,并且刚刚进入一个奖励关.在这个奖励关里,系统将依次随机抛出k次宝物,每次你都可以选择吃或者不吃(必 ...

  10. DE2资源集锦

    1.The School of Electrical and Computer Engineering (ECE) at the Georgia Institute of Technology:htt ...