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\),判断序列和是否非负. 对于询问\(( ...
随机推荐
- JavaSE---多线程---线程的控制
1.Java提供了一些工具方法,可以便捷控制线程的执行: 1.1 join Thread提供了让一个线程等待另一个线程执行完成的方法:join: 当某个程序的执行流中调用其他线程的join方法,该线程 ...
- youlexuan父类配置
<!-- 集中定义依赖版本号 --> <properties> <junit.version>4.12</junit.version> ...
- 【Vue】vue的双向绑定原理及实现
vue数据双向绑定是通过数据劫持结合发布者-订阅者模式的方式来实现的,那么vue是如果进行数据劫持的,我们可以先来看一下通过控制台输出一个定义在vue初始化数据上的对象是个什么东西. 代码: var ...
- 【Flutter学习】组件通信(父子、兄弟)
一,概述 flutter一个重要的特性就是组件化.组件分为两种状态,一种是StatefulWidget有状态组件,一种是StatelessWidget无状态组件. 无状态组件不能更新状态,有状态组件具 ...
- vue 学习中遇到的一些问题
Vue中data有return返回函数问题: 在简单的vue实例中会看到Vue实例中data属性展示方式如下: let app= newVue({ el:"#app", data: ...
- zabbix历史数据全部清楚
#这种方法会出现 监控项不可用的情况 1.停掉zabbix_server 2.重重名表 RENAME TABLE history to history_20180117; RENAME TABLE h ...
- [Repost] 探寻C++最快的读取文件的方案
作者:BYVoid(https://www.byvoid.com/zhs/blog/fast-readfile) 版权协议:CC BY-NC-SA 3.0 Unported 在竞赛中,遇到大数据时,往 ...
- MySQL教程和使用手册
MySQL 教程 MySQL 教程.MySQL 安装.MySQL 管理.MySQL PHP 语法.MySQL 连接.MySQL 创建数据库.MySQL 删除数据库.MySQL 选择数据库.MySQL ...
- MySQL 5.7配置文件
原文:http://www.voidcn.com/article/p-zrikccdi-hr.html # MySql5.7配置文件my.cnf设置[client]port = 3306socket ...
- JS-layui:百科
ylbtech-JS-layui:百科 layui,是一款采用自身模块规范编写的前端 UI 框架,遵循原生 HTML/CSS/JS 的书写与组织形式,门槛极低,拿来即用.其外在极简,却又不失饱满的内在 ...