这道题要看出来这个做法还是比较容易说一下细节

1.因为要用hash的区间值域来建树,而hash为了不冲突要开的很大,所以值域就会比较的大,不过这道题好的一点是没有修改,所以直接离散一下就会小很多

2.hash的时候多mod (' '     )

3.mod 的值可以稍微取大一点

 #include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; typedef long long ll; const ll maxn = ;
const ll mod = (ll)1e14 + ;
const ll p = ; ll a[maxn], b[maxn], pos[maxn], v[maxn], f[maxn], num = , n, m, k; ll int_get() {
ll x = ; char c = (char)getchar(); bool f = ;
while(!isdigit(c) && c != '-') c = (char)getchar();
if(c == '-') c = (char)getchar(), f = ;
while(isdigit(c)) {
x = x * + (int)(c - '');
c = (char)getchar();
}
if(f) x = -x;
return x;
} struct node {
ll data;
node *l, *r;
}e[maxn * ]; ll ne = ;
node* root[maxn]; void test(node* x, ll l, ll r) {
if(!x) return;
cout << l <<" "<< r << " "<< x-> data << endl;
if(l ^ r) {
ll mid = (l + r) >> ;
test(x-> l, l, mid), test(x-> r, mid + , r);
}
} void insert(node* &x, node* y, ll l, ll r, ll v) {
if(!x) x = e + ne ++;
if(l == r) x-> data = (y ? y-> data : ) + ;
else {
ll mid = (l + r) >> ;
if(v <= mid) {
if(y) x-> r = y-> r, y = y-> l;
insert(x-> l, y, l, mid, v);
}
else {
if(y) x-> l = y-> l, y = y-> r;
insert(x-> r, y, mid + , r, v);
}
}
} ll ask(node* x, node* t, ll l, ll r, ll v) {
if(!x) return ;
if(l == r) return x-> data - (t ? t-> data : );
else {
ll mid = (l + r) >> ;
if(v <= mid) {
if(t) t = t-> l;
return ask(x-> l, t, l, mid, v);
}
else {
if(t) t = t-> r;
return ask(x-> r, t, mid + , r, v);
}
}
} bool cmp(ll x, ll t) {
return b[x] < b[t];
} void pre(ll n) {
f[] = ;
for(ll i = ; i <= n; ++ i) f[i] = f[i - ] * p % mod;
} ll find(ll x) {
ll l = , r = num + ;
while(r - l > ) {
ll mid = (l + r) >> ;
if(b[pos[mid]] <= x) l = mid;
else r = mid;
}
return (x == b[pos[l]] ? v[pos[l]] : -);
} void read() {
n = int_get(), m = int_get(), k = int_get();
pre(n);
for(ll i = ; i <= n; ++ i) a[i] = int_get();
memset(b, , sizeof(b));
for(ll i = ; i <= k; ++ i) b[] = (b[] * p % mod + a[i]) % mod;
for(ll j = k + ; j <= n; ++ j)
b[j - k + ] = (((b[j - k] - a[j - k] * f[k - ] % mod) % mod + mod) % mod * p % mod + a[j]) % mod;
//for(ll i = 1; i <= n - k + 1; ++ i) cout << b[i] << endl;
//cout << endl;
for(int i = ; i <= n - k + ; ++ i) pos[i] = i;
sort(pos + , pos + + n - k + , cmp);
v[pos[]] = ++ num;
for(ll i = ; i <= n - k + ; ++ i) {
if(b[pos[i]] != b[pos[i - ]]) ++ num;
v[pos[i]] = num;
}
for(ll i = ; i <= n - k + ; ++ i) insert(root[i], root[i - ], , num, v[i]);
} void sov() {
while(m --) {
ll ls, rs;
ls = int_get(); rs = int_get(); rs -= k - ;
ll x = ;
for(ll i = ; i <= k; ++ i) {
ll s = int_get();
x = (x * p % mod + s ) % mod;
}
ll c = find(x);
if(c == - || rs < ls) printf("Yes\n");
else {
if(ask(root[rs], root[ls - ], , num, c) > ) printf("No\n");
else printf("Yes\n");
}
}
} int main() {
//freopen("test.in", "r", stdin);
//freopen("test.out", "w", stdout);
read();
//for(int i = 1; i <= n - k + 1; ++ i) test(root[i], 1, num), cout << endl;
sov();
}

bzoj 3207 可持久化线段树+hash的更多相关文章

  1. bzoj 3207 可持久化线段树

    首先因为固定询问长度,所以我们可以将整个长度为n的数列hash成长度为n-k+1的数列,每次询问的序列也hash成一个数,然后询问这个数是不是在某个区间中出现过,这样我们可以根据初始数列的权值建立可持 ...

  2. bzoj 3123 可持久化线段树启发式合并

    首先没有连边的操作的时候,我们可以用可持久化线段树来维护这棵树的信息,建立权值可持久化线段树,那么每个点继承父节点的线段树,当询问为x,y的时候我们可以询问rot[x]+rot[y]-rot[lca( ...

  3. bzoj 3524 可持久化线段树

    我们可以先离散化,然后建立权值的可持久化线段树,记录每个数出现的次数,对于区间询问直接判断左右儿子的cnt是不是大于(r-k+1)/2,然后递归到最后一层要是还是大于就有,否则不存在. 反思:挺简单一 ...

  4. BZOJ 2124等差子序列 线段树&&hash

    [题目描述 Description] 给一个 1 到 N 的排列{Ai},询问是否存在 1<=p1<p2<p3<p4<p5<…<pLen<=N(Len& ...

  5. BZOJ 2588: Spoj 10628. Count on a tree-可持久化线段树+LCA(点权)(树上的操作) 无语(为什么我的LCA的板子不对)

    2588: Spoj 10628. Count on a tree Time Limit: 12 Sec  Memory Limit: 128 MBSubmit: 9280  Solved: 2421 ...

  6. [BZOJ 3207] 花神的嘲讽计划Ⅰ【Hash + 可持久化线段树】

    题目链接:BZOJ - 3207 题目分析 先使用Hash,把每个长度为 k 的序列转为一个整数,然后题目就转化为了询问某个区间内有没有整数 x . 这一步可以使用可持久化线段树来做,虽然感觉可以有更 ...

  7. [BZOJ 3218] A + B Problem 【可持久化线段树 + 网络流】

    题目连接:BZOJ - 3218 题目分析 题目要求将 n 个点染成黑色或白色,那么我们可以转化为一个最小割模型. 我们规定一个点 i 最后属于 S 集表示染成黑色,属于 T 集表示染成白色,那么对于 ...

  8. BZOJ.4771.七彩树(可持久化线段树)

    BZOJ 考虑没有深度限制,对整棵子树询问怎么做. 对于同种颜色中DFS序相邻的两个点\(u,v\),在\(dfn[u],dfn[v]\)处分别\(+1\),\(dfn[LCA(u,v)]\)处\(- ...

  9. BZOJ.2653.[国家集训队]middle(可持久化线段树 二分)

    BZOJ 洛谷 求中位数除了\(sort\)还有什么方法?二分一个数\(x\),把\(<x\)的数全设成\(-1\),\(\geq x\)的数设成\(1\),判断序列和是否非负. 对于询问\(( ...

随机推荐

  1. visual Studio如何使用断点调试程序?

    1.在想要添加断点的地方右侧点击,点击成功后会出现红色原点. 2.启动程序,当进行到断点处时,程序会停止,然后可以看到一个黄色的小箭头在断点处 3.快捷键F10:进行下一句代码 4.快捷键F11:进入 ...

  2. python-列表list- 元组(tuple)- 集合(set)-字典(dict)-实例代码

    python 经常用的数据类型还有列表list- 元组(tuple)- 集合(set)-字典(dict),以下是这四种类型用法的简单示例. 1.列表list        [  ] 元素可变 , 有序 ...

  3. sql的分页

    public static string GetPageSql(string sql, int start, int end)        {            return string.Fo ...

  4. I/O 优化

    数字的读入优化: /* I/O * Au: GG */ #include <cstdio> #include <cstdlib> #include <cmath> ...

  5. uploadify的使用错误

    在看singwa的视频教程中,学习使用hui-admin模版,在使用uploadify插件上传图片中出现错误. ReferenceError: Can't find variable: $因为使用JQ ...

  6. python笔试做错的题目

    a = [1,2,3] b = a print(id(a),id(b),a == b) print(a,b) b = b + [1,2,3] print(a,b) print(id(a),id(b), ...

  7. 2.Prometheus安装部署

    环境准备 2台Linux操作系统(基于centos7) docker环境 配置 IP 角色 版本 192.168.229.139 prometheus-server 2.10 192.168.229. ...

  8. list列表操作(创建、增加、删除、取值)

    list ####(一)列表的创建[].追加(append,extend,insert).删除(remove.del.poop).修改 ##创建一个空列表.一个字符串列表.一个数字列表 lis0 = ...

  9. Oracle上课学习笔记<1>

    简单的select查询语句 1.select查询语句基本语法 使用两个关键字: select 指定要查询的字段和内容 from 从哪张表中查询 语法:select 字段名 from 表名; 三种不同的 ...

  10. 关于软件IntelliJ IDEA的使用技巧(四)

    二,IntelliJ IDEA的工具栏介绍 2,IntelliJ IDEA菜单栏 (9)Run运行 ✌1.Run'All Features in :src':运行scr中所有的特征 ✌2.Debug  ...