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]\) 的最大 ...
随机推荐
- gstreamer——文档/资源/使用
http://gstreamer.freedesktop.org/src/ http://gstreamer.freedesktop.org/data/doc/gstreamer/head/qt-gs ...
- 进程控制块PCB结构体 task_struct 描述
进程控制块,英文名(Processing Control Block),简称 PCB . 进程控制块是系统为了管理进程设置的一个专门的数据结构,主要表示进程状态. 每一个进程都对应一个PCB来维护进程 ...
- leetcode刷题3.无重复字符的最长子串
题目:给定一个字符串,找出不含有重复字符的 最长子串 的长度. 示例: 给定 "abcabcbb" ,没有重复字符的最长子串是 "abc" ,那么长度就是3. ...
- 《深度学习框架PyTorch:入门与实践》的Loss函数构建代码运行问题
在学习陈云的教程<深度学习框架PyTorch:入门与实践>的损失函数构建时代码如下: 可我运行如下代码: output = net(input) target = Variable(t.a ...
- linux系统怎么截图?linux系统中对指定区域进行截图的详细教程
windows系统的截图相当简单,方法也很多,但是linux下截图已经成为了一个老大难问题,在windows下用惯了qq截图,到了linux下没了qq,那要怎么办呢,prt sc sysrq 键全屏截 ...
- Kubernetes StatefulSets
StatefulSets对于需要以下一项或多项的应用程序非常有用. 稳定,唯一的网络标识符. 稳定,持久的存储. 有序,优雅的部署和缩放. 有序,优雅的删除和终止. 有序的自动滚动更新. POD Id ...
- Hessian与Spring整合
1.服务端与Spring的整合 1.1:web.xml中配置控制器 <servlet> <servlet-name>hessian</servlet-name> & ...
- asp.net web api history and how does it work?
https://blogs.msdn.microsoft.com/zxue/2012/11/07/what-is-asp-net-web-api-and-how-does-it-work/ https ...
- hive学习8(小案例1练习)
创建数据库 hive> create database feigu; hive> use feigu; 创建表 stg_job表 drop table if exists stg_job; ...
- python练习_sed替换
python练习_sed替换 需求: 做一个sed替换小程序,实现在windows下可以与实现linux中sed替换的功能 支持正则(re模块) 以下代码实现的功能与思路: 功能: (1)支持文件内容 ...