bzoj 3207 可持久化线段树+hash
这道题要看出来这个做法还是比较容易说一下细节
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的更多相关文章
- bzoj 3207 可持久化线段树
首先因为固定询问长度,所以我们可以将整个长度为n的数列hash成长度为n-k+1的数列,每次询问的序列也hash成一个数,然后询问这个数是不是在某个区间中出现过,这样我们可以根据初始数列的权值建立可持 ...
- bzoj 3123 可持久化线段树启发式合并
首先没有连边的操作的时候,我们可以用可持久化线段树来维护这棵树的信息,建立权值可持久化线段树,那么每个点继承父节点的线段树,当询问为x,y的时候我们可以询问rot[x]+rot[y]-rot[lca( ...
- bzoj 3524 可持久化线段树
我们可以先离散化,然后建立权值的可持久化线段树,记录每个数出现的次数,对于区间询问直接判断左右儿子的cnt是不是大于(r-k+1)/2,然后递归到最后一层要是还是大于就有,否则不存在. 反思:挺简单一 ...
- BZOJ 2124等差子序列 线段树&&hash
[题目描述 Description] 给一个 1 到 N 的排列{Ai},询问是否存在 1<=p1<p2<p3<p4<p5<…<pLen<=N(Len& ...
- 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 ...
- [BZOJ 3207] 花神的嘲讽计划Ⅰ【Hash + 可持久化线段树】
题目链接:BZOJ - 3207 题目分析 先使用Hash,把每个长度为 k 的序列转为一个整数,然后题目就转化为了询问某个区间内有没有整数 x . 这一步可以使用可持久化线段树来做,虽然感觉可以有更 ...
- [BZOJ 3218] A + B Problem 【可持久化线段树 + 网络流】
题目连接:BZOJ - 3218 题目分析 题目要求将 n 个点染成黑色或白色,那么我们可以转化为一个最小割模型. 我们规定一个点 i 最后属于 S 集表示染成黑色,属于 T 集表示染成白色,那么对于 ...
- BZOJ.4771.七彩树(可持久化线段树)
BZOJ 考虑没有深度限制,对整棵子树询问怎么做. 对于同种颜色中DFS序相邻的两个点\(u,v\),在\(dfn[u],dfn[v]\)处分别\(+1\),\(dfn[LCA(u,v)]\)处\(- ...
- BZOJ.2653.[国家集训队]middle(可持久化线段树 二分)
BZOJ 洛谷 求中位数除了\(sort\)还有什么方法?二分一个数\(x\),把\(<x\)的数全设成\(-1\),\(\geq x\)的数设成\(1\),判断序列和是否非负. 对于询问\(( ...
随机推荐
- python连接mongodb集群
一 安装模块pymongo pip3 install pymongo 二 创建一个MongoClient conn=MongoClient('mongodb://cbi:pass@ip1:20000, ...
- 【leetcode】316. Remove Duplicate Letters
题目如下: Given a string which contains only lowercase letters, remove duplicate letters so that every l ...
- 【IP】DHCP介绍
DHCP(Dynamic Host Configuration Protocol,动态主机配置协议)是一个局域网的网络协议,使用UDP协议工作, 主要有两个用途:给内部网络或网络服务供应商自动分配IP ...
- Oracle with as语法
with as优点 增加了sql的易读性,如果构造了多个子查询,结构会更清晰: 更重要的是:“一次分析,多次使用”,这也是为什么会提供性能的地方,达到了“少读”的目标 用法:给查询的语句起个别名 e. ...
- Qt installer framework学习
一.官网的介绍部分网址 http://doc.qt.io/qtinstallerframework/ifw-overview.html 二.安装界面介绍 2.1 安装界面流程 介绍>>选择 ...
- Mysql学习-安装与启动
安装mysqlrpm -ivh Mysql-server....rpm安装完成后,会提示/usr/bin/mysqladmin -u root password 'new-password'也可以用m ...
- Network基础(四):MAC地址表及邻居信息查看、配置接口速率及双工模式、配置交换机管理IP
一.MAC地址表及邻居信息查看 目标: 本例要求为修改计算机名并加入工作组: 查看交换机MAC地址表 查看CISCO设备邻居信息 方案: 网络拓扑,如下图所示. 步骤: 步骤一:查看交换机sw1的ma ...
- vue+express+mysql +node项目搭建
项目搭建前需要先安装node环境及mysql数据库. 1.利用vue-cli脚手架创建一个vue项目 a.全局安装 npm install -g vue-cli b.初始化项目 vue init we ...
- [CSP-S模拟测试41]题解
中间咕的几次考试就先咕着吧…… A.夜莺与玫瑰 枚举斜率.只考虑斜率为正且不平行于坐标轴的直线,最后把$ans\times 2$再$+1$即可. 首先肯定需要用$gcd(i,j)==1$确保斜率的唯一 ...
- 如何编写高性能的 javascript
一.Javascript代码执行效率1. DOM1.1 使用 DocumentFragment 优化多次 append说明:添加多个 dom 元素时,先将元素 append 到 DocumentFra ...