BZOJ 4059 [Cerc2012]Non-boring sequences(启发式分治)

【题目链接】 http://www.lydsy.com/JudgeOnline/problem.php?id=4059

【题目大意】

  一个序列被称为是不无聊的,仅当它的每个连续子序列存在一个独一无二的数字,
  即每个子序列里至少存在一个数字只出现一次。
  给定一个整数序列,请你判断它是不是不无聊的。

【题解】

  预处理每个元素上一次出现位置和下一个出现位置,
  我们发现对于一个子序列[L,R]来说,
  如果存在pre[i]<L&&nxt[i]>R那么这个子序列一定是满足条件的,
  否则就不满足,那么我们分治处理这个问题,
  从两边往中间寻找这个i,那么每次拆开的复杂度就是拆成的两个序列中较小的一个,
  所以这是一个逆启发式合并的过程,复杂度O(nlogn)

【代码】

#include <cstdio>
#include <algorithm>
#include <map>
using namespace std;
const int N=200010;
int n,T,a[N],pre[N],nxt[N];
bool check(int L,int R){
    if(L>=R)return 1;
    int l=L,r=R;
    for(int i=L;i<=R;i++){
        if(i&1){if(pre[l]<L&&nxt[l]>R)return(check(L,l-1)&&check(l+1,R));l++;}
        else{if(pre[r]<L&&nxt[r]>R)return(check(L,r-1)&&check(r+1,R));r--;}
    }return 0;
}
int main(){
    scanf("%d",&T);
    while(T--){
        scanf("%d",&n);
        map<int,int> M;
        for(int i=1;i<=n;i++){
            scanf("%d",&a[i]);
            nxt[M[a[i]]]=i;
            pre[i]=M[a[i]];
            M[a[i]]=i;
        }for(int i=1;i<=n;i++)nxt[M[a[i]]]=n+1;
        if(check(1,n))puts("non-boring");
        else puts("boring");
    }return 0;
}
时间: 2024-08-23 11:20:29

BZOJ 4059 [Cerc2012]Non-boring sequences(启发式分治)的相关文章

BZOJ 4059: [Cerc2012]Non-boring sequences ( )

要快速在一段子序列中判断一个元素是否只出现一次 , 我们可以预处理出每个元素左边和右边最近的相同元素的位置 , 这样就可以 O( 1 ) 判断. 考虑一段序列 [ l , r ] , 假如我们找到了序列中唯一元素的位置 p , 那我们只需检查 [ l , p - 1 ] & [ p + 1 , r ] 是否 non-boring 即可 . 如何检查 序列 [ l , r ] 呢 ? 假如从左往右或者从右往左找 , 最坏情况下是 O( n ) , 总时间复杂度会变成 O( n² ) ; 假如我们从

BZOJ 4059 Cerc2012 Non-boring sequences 线段树+扫描线

题目大意:定义一个序列为[不无聊的]当且仅当这个序列的任意一个区间都存在一个数只出现过一次,给定一个序列,要求判断这个序列是否是[不无聊的] 定义lasti表示第i个元素上一次出现的位置(第一次出现则为0),nexti表示第i个元素下一次出现的位置(最后一次出现则为n+1),那么这个元素能成为某个区间仅出现一次的数,当且仅当这个区间的左端点在[lasti+1,i]之间,右端点在[i,nexti?1]之间 我们可以将区间的左右端点放在二维平面上,那么一个元素产生的贡献是一个矩形,我们要确定的是所有

【BZOJ 4059】 (分治暴力|扫描线+线段树)

4059: [Cerc2012]Non-boring sequences Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 632  Solved: 227 Description 我们害怕把这道题题面搞得太无聊了,所以我们决定让这题超短.一个序列被称为是不无聊的,仅当它的每个连续子序列存在一个独一无二的数字,即每个子序列里至少存在一个数字只出现一次.给定一个整数序列,请你判断它是不是不无聊的. Input 第一行一个正整数T,表示有T组数据.每组数

启发式分治入门 Non-boring sequences UVA - 1608

参考自:https://blog.csdn.net/XY20130630/article/details/50635756 题意:一个序列被称为是不无聊的,仅当它的每个连续子序列存在一个独一无二的数字,即每个子序列里至少存在一个数字只出现一次.给定一个整数序列,请你判断它是不是不无聊的. 分析:预处理每个元素上一次出现位置和下一个出现位置, 我们发现对于一个子序列[L,R]来说, 如果存在pre[i]<L&&nxt[i]>R那么这个子序列一定是满足条件的, 否则就不满足,那么我

BZOJ 4059 Non-boring sequences

原来要从两边开始for啊.....感谢http://blog.csdn.net/XY20130630/article/details/50635756 #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> #include<map> #define maxn 200500 using namespace std; int t,n,a[maxn],aft[

BZOJ 2152 聪聪可可 (树上点分治)

题目地址:BZOJ 2152 找有多少对权值和为3的倍数的点.最简单的点分治. 代码如下: #include <iostream> #include <string.h> #include <math.h> #include <queue> #include <algorithm> #include <stdlib.h> #include <map> #include <set> #include <st

BZOJ 2726: [SDOI2012]任务安排( dp + cdq分治 )

考虑每批任务对后面任务都有贡献, dp(i) = min( dp(j) + F(i) * (T(i) - T(j) + S) ) (i < j <= N)  F, T均为后缀和. 与j有关的量只有t = dp(j) - F(i) * T(j) , 我们要最小化它. dp(j)->y, T(j)->x, 那么y = F(i) * x + t, 就是给一些点和一个斜率...然后最小化截距, 显然维护下凸包就可以了. 然后因为无比坑爹的出题人....时间可以为负数, 所以要用平衡树维护(

BZOJ 2287 POJ Challenge 消失之物 分治+背包

题目大意:给定n个物品,每个物品有一个体积,对于所有的1≤i≤n,1≤j≤m输出在不使用第i个物品的情况下装满大小为j的背包的方案数 我这傻逼居然真的去写了分治背包-- 第i个物品存在的时间为[1,i?1]和[i+1,n]两个区间 然后分治-- 时间复杂度O(n2logn) 黄学长我仰慕您 #include <vector> #include <cstdio> #include <cstring> #include <iostream> #include &

BZOJ 2716: [Violet 3]天使玩偶 [CDQ分治]

传送门 题意: 维护二维点集P,支持以下两个操作(1)插入点(x,y)(2)给定询问(x,y),求点集中离询问点最近的点距离定义为曼哈顿距离Dis(P1,P2)=|x1-x2|+|y1-y2|n,m<=500000x,y<=1000000 时间,$x$,$y$ $CDQ$分治里需要四个象限分类讨论,树状数组维护最大值 然后有两个象限是后缀和 然后跟$PoPoQQQ$学了一个神奇的技巧,树状数组加一个时间戳,就可以不用每次清空之前的操作了 #include <iostream> #i