BZOJ3745:[COCI2015]Norma
浅谈离线分治算法:https://www.cnblogs.com/AKMer/p/10415556.html
题目传送门:https://lydsy.com/JudgeOnline/problem.php?id=3745
我们对于所有区间,要么在分治的\(mid\)左边,要么在分治的\(mid\)右边,我们可以递归去处理。
所以我们只需要知道怎么快速统计经过\(mid\)的区间的答案即可。
我们从大到小枚举区间的左端点\(x\),然后计算\(\sum\limits_{y=mid+1}^{r}ans[x][y]\),\(ans[x][y]\)表示区间\([x,y]\)对答案的贡献。
假设区间\([x,mid]\)的最小值是\(a\),最大值\(b\)。
区间\([mid+1,q]\)的最小值比\(a\)大,最大值比\(b\)小。
区间\([q+1,p]\)的最小值比\(a\)大,最大值比\(b\)大,或者最小值比\(a\)小,最大值比\(b\)小。
区间\([p+1,r]\)的最小值小于\(a\),最大值大于\(b\)。
\(mn[i]\)表示\([mid+1,i]\)的最小值,\(mx[i]\)表示\([mid+1,i]\)的最大值。
当右端点落在这三段时,对答案的贡献分别计算。
第一段\([mid+1,q]\)对答案的贡献:
\(\sum\limits_{i=mid+1}^{q}ab(i-x+1)=\frac{ab(mid+2-x)(q-x+1)(q-mid)}{2}\)
第二段\([q+1,p]\)对答案的贡献:
假如这一段最小值比\(a\)大,最大值比\(b\)大:
\(\sum\limits_{i=q+1}^{p}a*mx[i]*(i-x+1)\)
化开得:\(a*(\sum\limits_{i=q+1}^{p}mx[i]*i-(x-1)\sum\limits_{i=q+1}^{p}mx[i])\)
令\(sum[1][y]\)表示\(\sum\limits_{i=mid+1}^{y}mx[i]*i\),\(sum[2][y]\)表示\(\sum\limits_{i=mid+1}^{y}mx[i]\)
那么对答案的贡献就可以表示为\(a*(sum[1][p]-sum[1][q]-(x-1)*(sum[2][p]-sum[2][q])\)
假如这一段最小值比\(a\)小,最大值比\(b\)小,同理可得:
令\(sum[3][y]\)表示\(\sum\limits_{i=mid+1}^{y}mn[i]*i\),\(sum[4][y]\)表示\(\sum\limits_{i=mid+1}^{y}mn[i]\)
对答案的贡献可以表示为\(b*(sum[3][p]-sum[3][q]-(x-1)*(sum[4][p]-sum[4][q]))\)
第三段对答案的贡献:
\(\sum\limits_{i=p+1}^{r}mn[i]*mx[i]*(i-x+1)\)
可以化为\(\sum\limits_{i=p+1}^{r}mx[i]*mn[i]*i-(x-1)*\sum\limits_{i=p+1}^{r}mn[i]*mx[i]\)
令\(sum[5][y]\)表示\(\sum\limits_{i=mid+1}^{y}mx[i]*mn[i]*i\),\(sum[6][y]\)表示\(\sum\limits_{i=mid+1}^{y}mn[i]*mx[i]\)
对答案的贡献可以表示为\(sum[5][r]-sum[5][p]-(x-1)*(sum[6][r]-sum[6][p])\)
所以只需要扫一遍就可以统计当前分治层的答案了。
时间复杂度:\(O(nlogn)\)
空间复杂度:\(O(n)\)
代码如下:
#include <cstdio>
#include <algorithm>
using namespace std;
#define sqr(x) (1ll*(x)*(x)%pps)
const int maxn=5e5+5,pps=1e9,inf=2e9;
int n,ans;
int sum[7][maxn];
int num[maxn],mn[maxn],mx[maxn];
int read() {
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
return x*f;
}
void clear(int pos) {
mn[pos]=inf,mx[pos]=-inf;
for(int i=1;i<7;i++)
sum[i][pos]=0;
}
int calc(int a,int b) {
int c=a+b,d=b-a+1;
if(c%2==0)c/=2;
else d/=2;
return 1ll*c*d%pps;
}
void solve(int l,int r) {
if(l==r) {
ans=(ans+sqr(num[l]))%pps;
return;
}
int mid=(l+r)>>1;
solve(l,mid),solve(mid+1,r);
clear(mid);
for(int i=mid+1;i<=r;i++) {
mn[i]=min(mn[i-1],num[i]);
mx[i]=max(mx[i-1],num[i]);
sum[1][i]=(sum[1][i-1]+1ll*mx[i]*i%pps)%pps;
sum[2][i]=(sum[2][i-1]+mx[i])%pps;
sum[3][i]=(sum[3][i-1]+1ll*mn[i]*i%pps)%pps;
sum[4][i]=(sum[4][i-1]+mn[i])%pps;
sum[5][i]=(sum[5][i-1]+1ll*mn[i]*mx[i]%pps*i%pps)%pps;
sum[6][i]=(sum[6][i-1]+1ll*mn[i]*mx[i]%pps)%pps;
}
int a=inf,b=-inf,limit1=mid,limit2=mid;
for(int x=mid;x>=l;x--) {
a=min(a,num[x]),b=max(b,num[x]);
while(mn[limit1+1]>=a&&limit1<r)limit1++;
while(mx[limit2+1]<=b&&limit2<r)limit2++;
int q=min(limit1,limit2),p=max(limit1,limit2);
ans=(ans+(1ll*a*b%pps*calc(mid+2-x,q-x+1)%pps))%pps;
if(q==limit2)ans=(ans+1ll*a*(sum[1][p]-sum[1][q]-1ll*(x-1)*(sum[2][p]-sum[2][q])%pps)%pps)%pps;
else ans=(ans+1ll*b*(sum[3][p]-sum[3][q]-1ll*(x-1)*(sum[4][p]-sum[4][q])%pps)%pps)%pps;
ans=(ans+sum[5][r]-sum[5][p]-1ll*(x-1)*(sum[6][r]-sum[6][p])%pps)%pps;
}
}
int main() {
n=read();
for(int i=1;i<=n;i++)
num[i]=read();
solve(1,n);ans=(ans+pps)%pps;
printf("%d\n",ans);
return 0;
}
BZOJ3745:[COCI2015]Norma的更多相关文章
- 【BZOJ3745】[Coci2015]Norma cdq分治
[BZOJ3745][Coci2015]Norma Description Input 第1行,一个整数N: 第2~n+1行,每行一个整数表示序列a. Output 输出答案对10^9取模后的结果. ...
- [BZOJ3745][COCI2015]Norma[分治]
题意 题目链接 分析 考虑分治,记当前分治区间为 \(l,r\) . 枚举左端点,然后发现右端点无非三种情况: 极大极小值都在左边; 有一个在左边; 极大极小值都在右边; 考虑递推 \(l\) 的同时 ...
- bzoj3745: [Coci2015]Norma
Description Input 第1行,一个整数N: 第2~n+1行,每行一个整数表示序列a. Output 输出答案对10^9取模后的结果. 预处理每个位置的数作为最小/大值向左延伸的最大距离, ...
- bzoj3745: [Coci2015]Norma 分治,单调队列
链接 bzoj 思路 首先\(\sum\limits_{i=1}^{n}\sum\limits_{j=1}^{n}\sum\limits_{k=i}^{j}max(a_k)\)可以用单调队列求解.参见 ...
- 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 ...
- bzoj 3745 [Coci2015]Norma——序列分治
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3745 如果分治,就能在本层仅算过 mid 的区间了. 可以从中间到左边地遍历左边,给右边两个 ...
- bzoj 3745: [Coci2015]Norma【分治】
参考:https://blog.csdn.net/lych_cys/article/details/51203960 真的不擅长这种-- 分治,对于一个(l,r),先递归求出(l,mid),(mid+ ...
- BZOJ3745 / SP22343 NORMA2 - Norma 分治,CDQ分治
要命的题目. 写法:分类讨论进行计算. 枚举过每一个\(mid\)的所有区间.对于左端点\(i∈[l, mid - 1]\),向左推并计算\([l,mid]\)范围内的最大\(/\)最小值. 然后右端 ...
- bzoj 3745: [Coci2015]Norma
Description Solution 考虑分治: 我们要统计跨越 \(mid\) 的区间的贡献 分最大值和最小值所在位置进行讨论: 设左边枚举到了 \(i\),左边 \([i,mid]\) 的最大 ...
随机推荐
- mongodb简介和特性
1.mongodb是基于文档的(BSON,类似json的键值对来存储),不是基于表格,易于水平扩展,将内部相关的数据放在一起能提高数据库的操作性能.如果你想新建一个新的文档类型,不用事先告诉数据库关于 ...
- Python与硬件学习笔记:蜂鸣器(转)
相信大家对蜂鸣器都不会陌生,很多产品和方案中都会用到蜂鸣器,大部分都是使用蜂鸣器来做提示或报警,比如按键按下.开始工作.工作结束或是故障等等.这里对单片机在蜂鸣器驱动上的应用作一下描述. 蜂鸣器的介绍 ...
- 【atcoder】All Your Paths are Different Lengths[arc102D](乱搞)
题目传送门:https://arc102.contest.atcoder.jp/tasks/arc102_b 这道题有点毒瘤啊,罚时上天.. 显然若$ l=2^n $那么就可以直接二进制拆分,但是如果 ...
- 【codevs1907】方格取数3(最大流最小割定理)
网址:http://codevs.cn/problem/1907/ 题意:在一个矩阵里选不相邻的若干个数,使这些数的和最大. 我们可以把它看成一个最小割,答案就是矩阵中的所有数-最小割.先把矩阵按国际 ...
- 【转载】Java类加载原理解析
Java类加载原理解析 原文出处:http://www.blogjava.net/zhuxing/archive/2008/08/08/220841.html 1 基本信息 摘要: 每个j ...
- Codeforces Round #200 (Div. 1) BCD
为了锻炼个人能力奋力div1 为了不做原题从200开始 B 两个电线缠在一起了 能不能抓住两头一扯就给扯分开 很明显当len为odd的时候无解 当len为偶数的时候 可以任选一段长度为even的相同字 ...
- JavaScript -- 节点操作, 事件触发, 表单伸缩
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- QT 创建对话框 Dialog 实例
1. 2. dialog.h 头文件 #ifndef DIALOG_H #define DIALOG_H #include <QDialog> QT_BEGIN_NAMESPACE cla ...
- Intel微处理结构.docx
1. 段寄存器 CS(代码段),代码段是一个存储器区域,这里保存微处理器使用的代码(程序和过程).代码段寄存器定义了存放代码的存储器段的起始地址.在实模式下工作时,它定义一个64KB存储器段的起始地址 ...
- Specify compute hosts with SSDs
scheduler_driver = nova.scheduler.filter_scheduler.FilterScheduler scheduler_available_filters = nov ...