题目链接:BZOJ - 3207

题目分析

先使用Hash,把每个长度为 k 的序列转为一个整数,然后题目就转化为了询问某个区间内有没有整数 x 。

这一步可以使用可持久化线段树来做,虽然感觉可以有更简单的做法,但是我没有什么想法...

代码

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <algorithm>
#include <cstring>
#include <cmath> using namespace std; const int MaxN = 200000 + 5, P = 233, Mod = 3371723, MaxNode = 8000000 + 5; int n, m, k, en, TL_Index, Node_Index;
int A[MaxN], B[MaxN], Root[MaxN], Lc[MaxNode], Rc[MaxNode], T[MaxNode]; struct HashNode
{
int Pos, TL;
HashNode *Next;
} H[MaxN], *Ph = H, *Hash[Mod + 5]; bool Cmp(int *AA, int x, int *AB, int y) {
for (int i = 0; i < k; ++i)
if (AA[x + i] != AB[y + i]) return false;
return true;
} void Insert(int &Now, int Last, int s, int t, int x) {
if (Now == 0) Now = ++Node_Index;
if (s == t) {
T[Now] = T[Last] + 1;
return;
}
int m = (s + t) >> 1;
if (x <= m) {
Rc[Now] = Rc[Last];
Insert(Lc[Now], Lc[Last], s, m, x);
}
else {
Lc[Now] = Lc[Last];
Insert(Rc[Now], Rc[Last], m + 1, t, x);
}
} int main()
{
scanf("%d%d%d", &n, &m, &k);
for (int i = 1; i <= n; ++i) scanf("%d", &A[i]);
en = n - k + 1;
int HN, TL_i;
HashNode *Now;
TL_Index = 0;
Node_Index = 0;
for (int i = 1; i <= en; ++i) {
HN = 0;
for (int j = i; j < i + k; ++j) {
HN = HN * P + A[j];
if (HN > Mod) HN %= Mod;
}
Now = Hash[HN];
TL_i = 0;
while (Now != NULL) {
if (Cmp(A, i, A, Now -> Pos)) {
TL_i = Now -> TL;
break;
}
Now = Now -> Next;
}
if (TL_i == 0) {
++Ph; Ph -> Pos = i;
Ph -> TL = TL_i = ++TL_Index;
Ph -> Next = Hash[HN]; Hash[HN] = Ph;
}
Insert(Root[i], Root[i - 1], 1, n, TL_i);
}
int l, r, s, t, mid, x, y;
for (int i = 1; i <= m; ++i) {
scanf("%d%d", &l, &r);
for (int j = 1; j <= k; ++j) scanf("%d", &B[j]);
HN = 0;
for (int j = 1; j <= k; ++j) {
HN = HN * P + B[j];
if (HN > Mod) HN %= Mod;
}
TL_i = 0;
Now = Hash[HN];
while (Now != NULL) {
if (Cmp(B, 1, A, Now -> Pos)) {
TL_i = Now -> TL;
break;
}
Now = Now -> Next;
}
if (TL_i == 0 || r - l + 1 < k) printf("Yes\n");
else {
r = r - k + 1;
x = Root[l - 1]; y = Root[r];
s = 1; t = n;
while (s != t) {
mid = (s + t) >> 1;
if (TL_i <= mid) {
x = Lc[x]; y = Lc[y];
t = mid;
}
else {
x = Rc[x]; y = Rc[y];
s = mid + 1;
}
}
if (T[y] - T[x] > 0) printf("No\n");
else printf("Yes\n");
}
}
return 0;
}

  

[BZOJ 3207] 花神的嘲讽计划Ⅰ【Hash + 可持久化线段树】的更多相关文章

  1. BZOJ 3207: 花神的嘲讽计划Ⅰ( hash + 可持久化线段树 )

    O(NK)暴力搞出所有子串的哈希值, 然后就对哈希值离散化建权值线段树, 就是主席树的经典做法了.总时间复杂度O(NK+(N+Q)logN) ----------------------------- ...

  2. 【BZOJ3207】花神的嘲讽计划I 可持久化线段树/莫队

    看到题目就可以想到hash 然后很自然的联想到可持久化权值线段树 WA:base取了偶数 这道题还可以用莫队做,比线段树快一些 可持久化线段树: #include<bits/stdc++.h&g ...

  3. bzoj 3207 花神的嘲讽计划Ⅰ(哈希法+主席树)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3207 [题意] 给定一个文本串,多次询问K长的模式串是否在文本[l,r]区间内出现. ...

  4. bzoj 3207 花神的嘲讽计划Ⅰ 主席树+hash

    花神的嘲讽计划Ⅰ Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3112  Solved: 1086[Submit][Status][Discuss] ...

  5. bzoj 3207: 花神的嘲讽计划Ⅰ

    Description 背景 花神是神,一大癖好就是嘲讽大J,举例如下: "哎你傻不傻的![hqz:大笨J]" "这道题又被J屎过了!!" "J这程序 ...

  6. BZOJ 3207 花神的嘲讽计划Ⅰ(函数式线段树)

    题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=3207 题意:给出一个数列,若干询问.每个询问查询[L,R]区间内是否存在某个长度为K的子 ...

  7. BZOJ 3207: 花神的嘲讽计划Ⅰ(莫队+哈希)

    传送门 解题思路 刚开始写了个莫队+哈希+\(map\)的\(O(n\sqrt(n)log(n)\)的辣鸡做法,\(T\)飞了.后来看了看别人博客发现其实并不用拿\(map\)当桶存那些哈希值.因为只 ...

  8. BZOJ_3207_花神的嘲讽计划Ⅰ_哈希+主席树

    BZOJ_3207_花神的嘲讽计划Ⅰ_哈希+主席树 Description 背景 花神是神,一大癖好就是嘲讽大J,举例如下: “哎你傻不傻的![hqz:大笨J]” “这道题又被J屎过了!!” “J这程 ...

  9. 【BZOJ3207】花神的嘲讽计划Ⅰ Hash+主席树

    [BZOJ3207]花神的嘲讽计划Ⅰ Description 背景 花神是神,一大癖好就是嘲讽大J,举例如下: “哎你傻不傻的![hqz:大笨J]” “这道题又被J屎过了!!” “J这程序怎么跑这么快 ...

随机推荐

  1. 【LeetCode】Best Time to Buy and Sell Stock

    Say you have an array for which the ith element is the price of a given stock on day i. If you were ...

  2. short a = 128, byte b = (byte)a 强制类型转换

    package 笔试; public class ShortToByte { /** * @param args */ public static void main(String[] args) { ...

  3. Java基础知识强化之集合框架笔记38:Set集合之Set集合概述和特点

    1. Set集合概述和特点 Collection            |--List                     有序(存储顺序和取出顺序一致),可重复            |--Se ...

  4. 'Service' object has no attribute 'process'

    在使用selenium+phantomjs时,运行总是出现错误信息: 'Service' object has no attribute 'process' 出现该错误的原因是未能找到可执行程序&qu ...

  5. PL/SQL 记录集合IS TABLE OF的使用

    在PL/SQL代码块中使用select into 赋值的话,有可能返回的是一个结果集.此时,如果使用基本类型或自定义的记录类型,将会报错. 因此,需要定义一个变量,是某种类型的集合.下面以一个基于表的 ...

  6. 国人编写的开源 .net Ioc 框架——My.Ioc 简介

    My.Ioc 是作者开发的一款开源 IoC/DI 框架,下载地址在此处.它具有下面一些特点: 高效 在实现手段上,My.Ioc 通过使用泛型.缓存.动态生成代码.延迟注册.尽量使用抽象类而非接口等方式 ...

  7. CSS Clip剪切元素实例

    一.实例1: .fixed { position: fixed; width: 382px; height: 100px; background: red; border: 1px solid blu ...

  8. HTML5+移动APP(1)

    前言: 介绍使用html5+(nativejs)和mui开发移动app(包括Android和iOs) HBuilder h5+开发app的环境,是一个对eclipse做了深度定的IDE. 官网: ht ...

  9. 关于SQL IO的一些资料

      前些天在做优化的时候发现一个有意思的现象,单纯的SQL执行很快,秒级返回,但是页面响应却很慢,一直在想这是为什么呢?有点怀疑服务器的IO有问题,想了想做了个实验,模拟了同样的场景,通过优化SQL将 ...

  10. KAFKA分布式消息系统[转]

    KAFKA分布式消息系统  转自:http://blog.chinaunix.net/uid-20196318-id-2420884.html Kafka[1]是linkedin用于日志处理的分布式消 ...