浅谈离线分治算法: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的更多相关文章

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

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

  2. [BZOJ3745][COCI2015]Norma[分治]

    题意 题目链接 分析 考虑分治,记当前分治区间为 \(l,r\) . 枚举左端点,然后发现右端点无非三种情况: 极大极小值都在左边; 有一个在左边; 极大极小值都在右边; 考虑递推 \(l\) 的同时 ...

  3. bzoj3745: [Coci2015]Norma

    Description Input 第1行,一个整数N: 第2~n+1行,每行一个整数表示序列a. Output 输出答案对10^9取模后的结果. 预处理每个位置的数作为最小/大值向左延伸的最大距离, ...

  4. bzoj3745: [Coci2015]Norma 分治,单调队列

    链接 bzoj 思路 首先\(\sum\limits_{i=1}^{n}\sum\limits_{j=1}^{n}\sum\limits_{k=i}^{j}max(a_k)\)可以用单调队列求解.参见 ...

  5. 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 ...

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

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

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

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

  8. BZOJ3745 / SP22343 NORMA2 - Norma 分治,CDQ分治

    要命的题目. 写法:分类讨论进行计算. 枚举过每一个\(mid\)的所有区间.对于左端点\(i∈[l, mid - 1]\),向左推并计算\([l,mid]\)范围内的最大\(/\)最小值. 然后右端 ...

  9. bzoj 3745: [Coci2015]Norma

    Description Solution 考虑分治: 我们要统计跨越 \(mid\) 的区间的贡献 分最大值和最小值所在位置进行讨论: 设左边枚举到了 \(i\),左边 \([i,mid]\) 的最大 ...

随机推荐

  1. Linux文件系统管理 常见命令df、du、fsck、dumpe2fs

    概述 Linux文件系统管理 常见命令df.du.fsck.dumpe2fs. 文件系统查看命令:df 通过 df 命令可以査看已经挂载的文件系统的信息包括设备文件名.文件系统总大小.已经使用的大小. ...

  2. DNS 域名解析原理

    域名解析过程 1.在浏览器中输入www.qq.com域名,操作系统会先检查自己本地的hosts文件是否有这个网址映射关系,如果有,就先调用这个IP地址映射,完成域名解析. 2.如果hosts里没有这个 ...

  3. linux+java+webdriver chrome handless无界面启动

    网上现有的解决方案要么是windows下的,要么是python的,搞了一天终于解决了,记录如下. 1 下载chrome linux版和对应版本的webdriver,我这里使用的是chrome66和ch ...

  4. oracle创建表空间 扩展表空间文件 修改表空间自动增长

    1. 创建表空间 create tablespace SIRM2  datafile 'D:\oracle\product\10.2.0\oradata\orcl\SIRM2.dbf' size 10 ...

  5. CCNA 课程 四

    Vlan基础: Vlan的作用:把物理上分割的用户,让他们逻辑上在一起. Vlan 范围: 0- 4095 0  4095 是保留的 不可以使用 1 cisco 本证vlan 标准vlan 1 -10 ...

  6. mysql创建定时执行存储过程任务实现订单定时关闭

    CREATE PROCEDURE `tableName`.`procedureName`() LANGUAGE SQL NOT DETERMINISTIC CONTAINS SQL SQL SECUR ...

  7. ML 感知机(Perceptrons)

    感知机 Perceptrons 学习Hinton神经网络公开课的学习笔记 https://class.coursera.org/neuralnets-2012-001 1 感知机历史 在19世纪60年 ...

  8. bzoj 1087 [SCOI2005]互不侵犯King 状态压缩dp

    1087: [SCOI2005]互不侵犯King Time Limit: 10 Sec  Memory Limit: 162 MB[Submit][Status][Discuss] Descripti ...

  9. Navicat 导入Excel与增加主键

    1.当你需要导入某Excel文件时,你必须把这个lxsl文件用Excel先打开(与其他软件的导入有点不太,其他会报错已占用之类的) 2.设置主键 当你打开你导入的Excel文件时,会显示无主键,需要你 ...

  10. TCP/IP 详解笔记

    最早的 TCP 协议文档是 RFC793. TCP 提供一种面向连接的.可靠的字节流服务. 面向连接容易理解,那么什么是字节流服务呢? 答:两个应用程序通过 TCP 连接交换 8 bit 字节构成的字 ...