https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1810

题目给出一个1~n的排列,问有多少连续区间。连续区间的定义为区间内元素排序后之间间隔为1。

对于一个区间[l,r],令mid=(l+r)/2,我们如果能在O(n)内求解出左端点在[l,mid],右端点在[mid+1,r]的连续区间数量,就可以将问题一分为二,递归求解[l,mid] [mid+1,r]。

现在来求解上面所说的这个子问题,首先默认i<j,有一个结论max[i~j]-min[i~j]==j-i时[i,j]是一个连续区间。所以我们维护两组从mid(mid+1)出发,向左(右)延伸的后(前)缀max和min数组。max[i~j]=max(max[i],max[j]),min同理。我们只要找到所有i,j组合使得结论式成立即可。

但是很明显朴素枚举是n^2的,我们能不能优化到n呢?

观察max[i~j],min[i~j],他们的来源有4种组合,因为是两两对应的,我们其中两个组合来讨论,另外两个可以同理推导。

第一种是max和min都来自mid左边。有max[i]-min[i]=j-i,推导得 j=max[i]-min[i]+i,只要枚举左半边的i,并判断j是否合法即可。

第二种是max来自左边min来自右边。有max[i]-min[j]=j-i,推导得max[i]+i=min[j]+j,枚举每个max[i]+i,计算有多少合法的j符合即可。说得轻松,这个j的数量怎么计算呢?

我们发现max想要来自左边,需要满足max[i]>max[j],同理min[i]>min[j]。同时我们发现从中心向外发散,max递增,min递减,也就是从中心向左枚举i的过程中max限制越来越宽松,min越来越严格。我们lp,rp来维护右半区间中满足当前i的[mid+1,r]的j的子窗口,可以预见的是随着i--,这个窗口会向右尺取。我们每加入一个j,就让一个计数数组中的cnt[min[j]+j]++,每剔除一个j则相应减减。对于每个i,完成尺取后,ans+=cnt[max[i]+i]。

对应搞定剩下两个情况后这题就理论AC了,剩下还有一些细节,比如l==r的处理,cnt数组的复原处理,输入挂优化什么的,搞搞就AC了。

#include <iostream>
#include <cmath>
#include <algorithm>
#include <map>
#include <cstring>
#define LL long long
using namespace std;
const LL N = ;
int num[N],n;
LL ans;
int mx[N], mi[N];
int read() {
char ch;
for (ch = getchar(); ch<'' || ch>''; ch = getchar());
int x = ch - '';
for (ch = getchar(); ch >= ''&&ch <= ''; ch = getchar()) x = x * + ch - '';
return x;
}
void make_pre(LL l, LL r, LL mid)
{
mi[mid] = mx[mid] = num[mid];
mi[mid + ] = mx[mid + ] = num[mid + ];
for (int i = mid - ; i >= l; i--)
{
mx[i] = max(num[i], mx[i + ]);
mi[i] = min(num[i], mi[i + ]);
}
for (int i = mid + ; i <= r; i++)
{
mx[i] = max(num[i], mx[i - ]);
mi[i] = min(num[i], mi[i - ]);
}
}
struct tong
{
LL cnt[N * ];
void clear()
{
memset(cnt, , sizeof(cnt));
}
void setZero(LL num)
{
cnt[num + N] = ;
}
void add(LL num,int v)
{
cnt[num + N]+=v;
}
LL query(LL num)
{
return cnt[num+N];
}
}cnt; void solve(LL l, LL r)
{ int temp = ans;
LL mid = (l + r) /;
make_pre(l, r, mid);
//same i
for (int i = l; i <= mid; i++)
{
int nj = mx[i] - mi[i]+i;
if (nj>mid&&nj<=r&&mx[i]>mx[nj] && mi[i]<mi[nj]) ans++;
}
//same j
for (int i = mid+; i <= r; i++)
{
int nj = mx[i] - mi[i]-i;
nj = -nj;
if (nj<=mid&&nj>=l&&mx[i]>mx[nj] && mi[i]<mi[nj]) ans++;
}
//dif mx[i],mi[j]
LL pl = mid+,pr=mid+;
for (int i = mid; i>=l; i--)
{
while (pr <= r&&mx[pr] < mx[i])cnt.add(mi[pr] + pr,),pr++;
while (pl < pr&&mi[pl] > mi[i])cnt.add(mi[pl] + pl, -), pl++;
//if (cnt.query(mx[i] + i) < 0) cout << l << ' ' << r << ' ' << pl << endl;
ans += cnt.query(mx[i] + i);
}
while (pl < pr)cnt.setZero(mi[pl]+pl),pl++;
//dif mi[i],mx[j]
pl = mid , pr = mid;
for (int i=mid+; i<=r; i++)
{
while (pr >=l&&mx[pr] < mx[i])cnt.add(mi[pr] - pr, ), pr--;
while (pl > pr&&mi[pl] > mi[i])cnt.add(mi[pl] - pl, -), pl--;
//if(cnt.query(mx[i] - i)<0)cout << l << ' ' << r << ' ' << pl << endl;
ans += cnt.query(mx[i] - i);
}
while (pl > pr)cnt.setZero(mi[pl] - pl), pl--;
//cout << ans - temp << ' ' << l << ' ' << r << endl;
if (l == r)return;
solve(l, mid);
solve(mid + , r);
}
int main()
{
//cin.sync_with_stdio(false);
n = read();
for (int i = ; i < n; i++)num[i]=read();
ans = ;
cnt.clear();
solve(, n - );
printf("%lld\n", ans+n); return ;
}

51Nod 1810 连续区间的更多相关文章

  1. 51NOD 1810 连续区间 分治 区间计数

    1810 连续区间 基准时间限制:1.5 秒 空间限制:131072 KB 分值: 80     区间内所有元素排序后,任意相邻两个元素值差为1的区间称为“连续区间” 如:3,1,2是连续区间,但3, ...

  2. 51nod 1094 和为k的连续区间【前缀和/区间差/map】

    1094 和为k的连续区间 基准时间限制:1 秒 空间限制:131072 KB 分值: 10 难度:2级算法题  收藏  关注 一整数数列a1, a2, ... , an(有正有负),以及另一个整数k ...

  3. 51Nod 1094 和为k的连续区间 | 水

    Input示例 6 10 1 2 3 4 5 6 Output示例 1 4 #include "cstdio" #include "algorithm" #in ...

  4. 51Nod 1094 和为k的连续区间

    #include <iostream> #include <algorithm> #include <cstring> using namespace std; t ...

  5. 51Nod 和为k的连续区间

    一整数数列a1, a2, ... , an(有正有负),以及另一个整数k,求一个区间[i, j],(1 <= i <= j <= n),使得a[i] + ... + a[j] = k ...

  6. 【51Nod 1244】莫比乌斯函数之和

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1244 模板题... 杜教筛和基于质因子分解的筛法都写了一下模板. 杜教筛 ...

  7. 51Nod 1268 和为K的组合

    51Nod  1268  和为K的组合 1268 和为K的组合 基准时间限制:1 秒 空间限制:131072 KB 分值: 20 难度:3级算法题 给出N个正整数组成的数组A,求能否从中选出若干个,使 ...

  8. 51Nod 1428 活动安排问题

    51Nod   1428  活动安排问题 Link: http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1428 1428 活 ...

  9. 51Nod 1278 相离的圆

    51Nod 1278 相离的圆 Link: http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1278 1278 相离的圆 基 ...

随机推荐

  1. jsoup对 HTML 文档的解析和操作

    本文手动转载自http://www.cnblogs.com/chenying99/archive/2013/01/04/2844615.html,仅根据个人需要对实用部分进行转载,详细请阅读原文. j ...

  2. topcoder srm 705 div1 -3

    1.设有一个字母表$T$,对于一个单词$w$,若对于任意的$0\leq i< |w|-1$,满足$w_{i}$在$T$中的排名小于等于$w_{i+1}$在$T$中的排名,则称$s$在$T$中是合 ...

  3. eMMC应用教程:关于RPMB的应用【转】

    本文转载自:https://blog.csdn.net/youdianhai/article/details/51246379 RPMB的意思是Replay Protected Memory Bloc ...

  4. linux内核中的vgaarb是什么?

    答: vga仲裁器(vga arbiter),是内核中的一个模块

  5. python 之 条件语句

    python 编程语言指定任何非0和非空(null)值为true, 0或者null为false. python 编程中if语句用于控制程序的执行,基本形式为: if 判断条件: 执行语句…… else ...

  6. class类的使用

    我们在ES5中经常使用方法或者对象去模拟类的使用,虽然可以实现功能,但是代码并不优雅,ES6为我们提供了类的使用.需要注意的是我们在写类的时候和ES5中的对象和构造函数要区分开来,不要学混了. 类的声 ...

  7. GPIO实验之c语言

    上一章节进行实验使用的是汇编进行编程的,本次实验是使用c语言进行编写的. (1)点亮一个led灯   1)启动文件:    crt.S   .text   .global _start   _star ...

  8. dRMT: Disaggregated Programmable Switching, SIGCOMM17

    Reference: dRMT, SIGCOMM 2017 今年的SIGCOMM17会议上,Cisco System和MIT的团队针对RMT模型现有的问题,合作发表了这篇"dRMT: Dis ...

  9. Python学习 day04打卡

    今天学习的主要内容: 一,列表 1,列表的介绍 列表是python的基础数据类型之一,其他编程语音也有类似的数据类型.例如:JS 中的数组Java中的数组等等. 它是以[]括起来,每个元素用',隔开而 ...

  10. 批量Excel数据导入Oracle数据库 导入excel错误:外部表不是预期的格式 解决方案

    在asp.net网站中导出Excel文件后,再把文件导入到数据库中. 读取Excel文件时,打开连接出错. 错误为:外部表不是预期的格式 解决:检查了一下,导出的Excel是标准文件不是html,没错 ...