题目链接: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. Android 动画之RotateAnimation应用详解

    android中提供了4中动画: AlphaAnimation 透明度动画效果 ScaleAnimation 缩放动画效果 TranslateAnimation 位移动画效果 RotateAnimat ...

  2. cocos2d&amp;cocos2dx学习资源

    汇总一下自己学习Cocos2d和cocos2dx认为比較好的一些资源: 书籍: <iPhone&iPad cocos2d游戏开发实战> Steffen Itterheim < ...

  3. UITableView学习总结

    一.UITableView概述 UITableView继承自UIScrollView,可以表现为Plain和Grouped两种风格. UITableView有两个Delegate分别为:dataSou ...

  4. ArcEngine:栅格分级渲染

    ArcEngine对矢量数据进行风格化实在是得心应手,同样的对于栅格图像也能进行风格化!以前没接触过,今天正好需要,做出了栅格图像的渲染!下面实现的思路: 1.定义渲染的一系列接口 2.判断图像是否建 ...

  5. [O] SQLite数据库报错:no such column

    在SQLite数据库创建语句增加列,运行后报错:no such column 在语法规范的前提下,即 //SQLite数据库创建,逗号与空格须严格 String CREATE_NOTE = " ...

  6. C# 异步操作

    在程序中,普通的方法是单线程的.但中途如果有大型的操作,比如读取大文件,大批量操作数据库,网络传输等,都会导致程序阻塞,表现在界面上就是程序卡或者死掉,界面元素不动了,不响应了.C#异步调用很好的解决 ...

  7. UOJ222 NOI2016 区间 线段树+FIFO队列

    首先将区间按长度排序后离散化端点(这里的“长度”指的是离散化之前区间的实际长度) 然后模拟一个队列,区间按排好的顺序依次进入,直到某个点被覆盖了M次.之后依次出队,直到所有点都被覆盖小于M次 修改和询 ...

  8. codeforces 165D.Beard Graph 解题报告

    题意: 给一棵树,树的每条边有一种颜色,黑色或白色,一开始所有边均为黑色,有两个操作: 操作1:将第i条边变成白色或将第i条边变成黑色. 操作2 :询问u,v两点之间仅经过黑色变的最短距离. 树链剖分 ...

  9. 【BZOJ1036】【树链剖分】树的统计Count

    Description 一 棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. ...

  10. 【NOI2004】郁闷的出纳员

    [问题描述] OIER公司是一家大型专业化软件公司,有着数以万计的员工.作为一名出纳员,我的任务之一便是统计每位员工的工资.这本来是一份不错的工作,但是令人郁闷的是,我们的老板反复无常,经常调整员工的 ...