【问题描述】
    长度为n 的一串项链,每颗珠子是K 种颜色之一。第i 颗与第i-1,i+1 颗珠子相邻,第n 颗与第1 颗也相邻。
    切两刀,把项链断成两条链。要求每种颜色的珠子只能出现在其中一条链中。
    求方案数量。
【输入】
    输入文件名为(neck.in)。
    第一行两个数n,K。
    第二行n 个数,第i 个数代表第i 颗珠子的颜色。
【输出】
    输出文件名为(neck.out)。
    一行一个数表示答案。
样例输入

neck.in
    4 3
    1 2 3 3

样例输出

neck.out
    3
 【数据范围与约定】
    对于20%的数据:n,K<= 10
    对于40%的数据:n,K<=1000
    对于100%的数据:n,K<= 1000000


题解:

称一个位置的后继为这个位置之后第一个与其颜色相同的位置。(i-1,i)表示i和i-1之间的间隔。

如果一个位置有后继,那么将这个位置的权值加上一个较大的随机整数,后继的权值减去这个整数。(此处用HASH实现)

设权值的前缀和为sum。

设i,j为两个位置,那么如果sum[i]=sum[j],就认为(i-1,i),(j-1,j)可以成为一对切割位置。

因为如果一个数和它的后继被分到不同两段,就会对一个sum产生贡献。

复杂度O(nlogn)

代码如下:

#include<bits/stdc++.h>
#define U unsigned
#define N 1100000
#define seed 19260817
using namespace std;
U long long pw[N],val[N],sum[N];
long long ans;
int n,K,tot,top,last[N],a[N],st[N],nex[N],vis[N];
int main(){
freopen("neck.in","r",stdin);
freopen("neck.out","w",stdout);
scanf("%d%d",&n,&K);
pw[]=;
for(int i=;i<=n;i++) pw[i]=pw[i-]*seed;
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
if(last[a[i]]){
val[last[a[i]]]+=pw[++tot];
val[i]-=pw[tot];
}
last[a[i]]=i;
}
for(int i=;i<=n;i++) sum[i]=sum[i-]+val[i];
sort(sum+,sum+n+);
int now=;
for(int i=;i<=n;i++){
now++;
if(i==n||sum[i]!=sum[i+])
ans+=(long long)now*(now-)/,now=;
}
printf("%d",ans);
return ;
}
#include<bits/stdc++.h>
#define U unsigned
#define N 1100000
#define seed 19260817
using namespace std;
U long long pw[N],val[N],sum[N];
long long ans;
int n,K,tot,top,last[N],a[N],st[N],nex[N],vis[N];
int main(){
freopen("neck.in","r",stdin);
freopen("neck.out","w",stdout);
scanf("%d%d",&n,&K);
pw[]=;
for(int i=;i<=n;i++) pw[i]=pw[i-]*seed;
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
if(last[a[i]]){
val[last[a[i]]]+=pw[++tot];
val[i]-=pw[tot];
}
last[a[i]]=i;
}
for(int i=;i<=n;i++) sum[i]=sum[i-]+val[i];
sort(sum+,sum+n+);
int now=;
for(int i=;i<=n;i++){
now++;
if(i==n||sum[i]!=sum[i+])
ans+=(long long)now*(now-)/,now=;
}
printf("%d",ans);
return ;
}

[BZOJ4382][POI2015]Podział naszyjnika (神奇HASH)的更多相关文章

  1. BZOJ4382 : [POI2015]Podział naszyjnika

    对于每种颜色,可以发现可以切的位置被分割成了若干段独立的区域. 给每个区域一个编号,将$m$种颜色的情况当成字符串来看,如果两个切口的字符串完全匹配,那么可以在这里切两刀. 可以构造hash函数,通过 ...

  2. 【BZOJ4382】[POI2015]Podział naszyjnika 堆+并查集+树状数组

    [BZOJ4382][POI2015]Podział naszyjnika Description 长度为n的一串项链,每颗珠子是k种颜色之一. 第i颗与第i-1,i+1颗珠子相邻,第n颗与第1颗也相 ...

  3. @bzoj - 4382@ [POI2015] Podział naszyjnika

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 长度为 n 的一串项链,每颗珠子是 k 种颜色之一. 第 i 颗 ...

  4. POI2015题解

    POI2015题解 吐槽一下为什么POI2015开始就成了破烂波兰文题目名了啊... 咕了一道3748没写打表题没什么意思,还剩\(BZOJ\)上的\(14\)道题. [BZOJ3746][POI20 ...

  5. [Poi2015]

    [POI2015]Łasuchy 一看以为是sb题 简单来说就是每个人获得热量要尽量多 不能找别人 首先这道题好像我自己找不到NIE的情况 很容易想到一个优化 如果一个数/2>另一个数 那么一定 ...

  6. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  7. Codeforces.700E.Cool Slogans(后缀自动机 线段树合并 DP)

    题目链接 \(Description\) 给定一个字符串\(s[1]\).一个字符串序列\(s[\ ]\)满足\(s[i]\)至少在\(s[i-1]\)中出现过两次(\(i\geq 2\)).求最大的 ...

  8. Luogu3587[POI2015]POD - hash + 单调队列

    Solution 还是去看了题解. 感谢大佬的博客→  题解传送门 是一道思路比较新的题. 搞一个前缀和, 记录前 $i$ 个位置每种颜色的出现次数, 如果位置 $i$ 是 颜色 $a[i]$ 的最后 ...

  9. BZOJ 3790 神奇项链 hash/后缀自动机+贪心

    Description 母亲节就要到了,小 H 准备送给她一个特殊的项链.这个项链可以看作一个用小写字母组成的字符串,每个小写字母表示一种颜色. 为了制作这个项链,小 H 购买了两个机器.第一个机器可 ...

随机推荐

  1. RFM - Customer Level Data

    Introduction ## Warning: package 'DT' was built under R version 3.5.2 RFM (recency, frequency, monet ...

  2. 假设检验总结以及如何用python进行假设检验(scipy)

    几种常见的假设检验总结如下: 假设检验名称 Z检验 t检验 χ2检验 F检验 原假设 H0: μ≥μ0        H0: μ≤μ0        H0: μ=μ0  (比较样本和总体均值)     ...

  3. AttributeError: module ‘select’ has no attribute 'epoll’

    场景:mac 下导入的 ‘select’ 包 import select,然后在 主函数 中创建的 epoll 对象 epl = select.epoll(),运行报错如下 Traceback (mo ...

  4. 【BigData】Java基础_冒泡排序

    1.实现需求 根据已经存在的数组,使用冒泡排序将数组中的元素排序后输出. 2.代码 package cn.test.logan.day02; /** * 冒泡排序在数组上的实现 * @author Q ...

  5. Expression Atlas

    网页  https://www.ebi.ac.uk/gxa/home 文档  https://www.ebi.ac.uk/gxa/help/index.html

  6. java基础之 内部类 & 嵌套类

    参考文档: 内部类的应用场景 http://blog.csdn.net/hivon/article/details/606312 http://wwty.iteye.com/blog/338628 定 ...

  7. 冰多多团队-第一次Scrum例会

    冰多多团队-第一次Scrum会议 注:由于对课程要求的不熟悉,所以本文档为周会后的补充总结文档 会议基本情况 会议时间:3.28 19:00 - 19:30 会议地点:新主楼F座2楼沙发休息处 工作情 ...

  8. Oracle之clob字段不能union的问题

    原因:由于clob类型字段不能使用group by函数,而union中需要使用group by过滤掉重复纪录: 解决方法:union可以改为union all.

  9. 动态查找之二叉树查找 c++实现

    算法思想 二叉搜索树(又称二叉查找树或二叉排序树)BST树 二叉查找树 二叉查找树,也称二叉搜索树,或二叉排序树.其定义也比较简单,要么是一颗空树,要么就是具有如下性质的二叉树: (1)若任意节点的左 ...

  10. SELECT语句中的for update的用法(锁的运用)

    回复1:一般FOR UPDATE用在PL/SQL的游标里,它的作用就是一个行级锁(对游标里所有的记录),如果其他进程要更新这个游标行级锁里的记录,就必须等待当前进程的COMMIT或者回滚. 该语句用来 ...