题目链接: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. StrutsPrepareAndExecuteFilter(转)

    http://www.iteye.com/topic/829843  一.概述 Struts2的核心是一个Filter,Action可以脱离web容器,那么是什么让http请求和action关联在一起 ...

  2. 数据库中存储日期的字段类型究竟应该用varchar还是datetime ?

    背景: 前段时间在百度经验看到一篇文章<如何在电脑右下角显示你(爱人)的名字>,之前也听过这个小技巧,但没真正动手设置过.所以出于好奇就实践了一下. 设置完成后的效果例如以下.右下角的时间 ...

  3. golang 学习笔记

    golan 声明的变量必须要用到? 语法 a,b:=2323; b为 bool 类型 结构体的赋值 需要用到逗号分隔字段 并且最后一个字段后也必须加上逗号 这和 JavaScript 的对象不一样哦 ...

  4. 京东手机webapp商城

    http://bases.wanggou.com/mcoss/mportal/show?tabid=2&ptype=1&actid=1562&tpl=3&pi=1&am ...

  5. android 25 跨进程启动activity

    跨进程启动activity,启动系统预定义的activity就是跨进程的. client项目启动server项目的activity. clientActivity.java package com.s ...

  6. [Android 中级]Voip之CSipSimple类库的编绎

    CSipSimple是什么?是一款基于pjsip的Android客户端,相信想要研究VOIP通讯的朋友一定不会陌生,这里我就把如何编译CSipSimple写下来. 首先从CSipSimple官方网站上 ...

  7. iOS-iPad开发之popoverController使用介绍

    iOS-iPad开发之popoverController使用介绍 iOS开发UI篇-popoverController使用注意 iOS SDK:自定义Popover(弹出窗口) 实现的简单例子: // ...

  8. linux device driver —— 环形缓冲区的实现

    还是没有接触到怎么控制硬件,但是在书里看到了一个挺巧妙的环形缓冲区实现. 此环形缓冲区实际为一个大小为bufsize的一维数组,有一个rp的读指针,一个wp的写指针. 在数据满时写进程会等待读进程读取 ...

  9. shell脚本实现冒泡排序 分类: 学习笔记 linux ubuntu 2015-07-10 14:16 79人阅读 评论(0) 收藏

    手动输入一行字符串,并对其排序. 脚本如下: #!/bin/bash #a test about sort echo "please input a number list" re ...

  10. 禁止鼠标多次点击选中div中的文字

    <!DOCTYPE html><html><head><meta charset="utf-8"><title>Fire ...