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]\) 的最大 ...
随机推荐
- jQuery消息提示框插件Tipso
在线演示 本地下载
- jsonp跨域总结
同源限制: 浏览器不允许一个域的脚本请求另一个域的文档,通俗来说,不允许访问不同协议.不同域名或不同端口的文档 跨域处理方法: 1.jsonp 前提: 浏览器虽然对ajax请求做出了限制,但scrip ...
- 20145239杜文超《网络对抗》- Web基础
20145239杜文超<网络对抗>- Web基础 基础问题回答 1.什么是表单? 表单是一个包含表单元素的区域. 表单元素是允许用户在表单中(比如:文本域.下拉列表.单选框.复选框等等)输 ...
- cocos2dx打飞机项目笔记七:各种回调:定时器schedule、普通回调callFunc、菜单回调menu_selector、事件回调event_selector
各种回调函数的定义: typedef void (CCObject::*SEL_SCHEDULE)(float); typedef void (CCObject::*SEL_CallFunc)(); ...
- windows 服务安装报错
使用windows服务开发的定时任务,在win7上都运行良好,在windows server 2008上运行报错,报错信息如下 错误应用程序名称: GCWindowsService.exe,版本: 1 ...
- INSPIRED启示录 读书笔记 - 第27章 合理运用瀑布式开发方法
瀑布式开发方法的基本原则 1.采用阶段式开发:软件开发过程被事先分成固定的几个阶段,撰写书面的需求说明文档.设计高层软件架构.设计低层细节.编写代码.测试.部署 2.采用阶段式评审:每个阶段结束后,对 ...
- 为什么原生的servlet是线程不安全的而Struts2是线程安全的?
因为原生的servlet在整个application生命周期中,只在初次访问的时候实例化一次,以后都不会再实例化,只会调用Server方法进行响应,所以如果在servlet类中定义成员变量,那么就会让 ...
- Spark基本概念快速入门
Spark集群 一组计算机的集合,每个计算机节点作为独立的计算资源,又可以虚拟出多个具备计算能力的虚拟机,这些虚拟机是集群中的计算单元.Spark的核心模块专注于调度和管理虚拟机之上分布式计算任务 ...
- 汇编笔记 RETF
assume cs:code stack segment db 16 dup(0) stack ends code segment start: mov ax,stack;将定义字形数据送入AX mo ...
- 应用程序.f/q(f了个墙)
1.20180414: 前两天,发现 CnFast不能用了,也没管它 以为过两天会好. 今天发现还是不能用,上网随便baidu下,看到个帖子:有人用加速精灵吗 现在登陆不了了 哪位大佬知道怎么回事啊[ ...