BZOJ4028 HEOI2015公约数数列(分块)
前缀gcd的变化次数是log的,考虑对每一种gcd查询,问题变为查询一段区间是否存在异或前缀和=x/gcd。
无修改的话显然可以可持久化trie,但这玩意实在没法支持修改。于是考虑分块。
对于每一块将其中所有块内异或前缀和排序。查询时先看这块与上一块相比gcd有没有变化,如果有对其中每个位置暴力查询,否则在排序后的数组中二分。修改时暴力改每一块的前缀gcd及异或和,被修改的块暴力重构排序数组即可。
- #include<iostream>
- #include<cstdio>
- #include<cmath>
- #include<cstdlib>
- #include<cstring>
- #include<algorithm>
- using namespace std;
- #define N 100010
- #define ll long long
- ll read()
- {
- ll x=,f=;char c=getchar();
- while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
- while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
- return x*f;
- }
- int n,m,a[N];
- int block_size,block_tot,L[N],R[N],pos[N];
- int gcd_pre[N],xor_pre[N],gcd_block[N],xor_block[N];
- struct data
- {
- int x,i;
- bool operator <(const data&a) const
- {
- return x<a.x||x==a.x&&i<a.i;
- }
- }v[N];
- int gcd(int n,int m){return m==?n:gcd(m,n%m);}
- int main()
- {
- #ifndef ONLINE_JUDGE
- freopen("bzoj4028.in","r",stdin);
- freopen("bzoj4028.out","w",stdout);
- const char LL[]="%I64d\n";
- #else
- const char LL[]="%lld\n";
- #endif
- n=read();
- for (int i=;i<=n;i++) a[i]=read();
- block_size=sqrt(n);block_tot=(n-)/block_size+;
- for (int i=;i<=block_tot;i++)
- L[i]=(i-)*block_size+,R[i]=min(n,i*block_size);
- for (int i=;i<=block_tot;i++)
- {
- for (int j=L[i];j<=R[i];j++)
- gcd_block[i]=gcd(gcd_block[i],a[j]),
- xor_block[i]=xor_block[i]^a[j],
- pos[j]=i;
- gcd_pre[i]=gcd(gcd_pre[i-],gcd_block[i]),xor_pre[i]=xor_pre[i-]^xor_block[i];
- }
- for (int i=;i<=block_tot;i++)
- {
- v[L[i]].x=a[L[i]],v[L[i]].i=L[i];
- for (int j=L[i]+;j<=R[i];j++)
- v[j].x=v[j-].x^a[j],v[j].i=j;
- sort(v+L[i],v+R[i]+);
- }
- m=read();
- while (m--)
- {
- char c=getchar();
- while (c<'A'||c>'Z') c=getchar();
- if (c=='M')
- {
- int p=read()+,x=read();a[p]=x;p=pos[p];
- gcd_block[p]=xor_block[p]=;
- for (int i=L[p];i<=R[p];i++)
- gcd_block[p]=gcd(gcd_block[p],a[i]),
- xor_block[p]=xor_block[p]^a[i];
- for (int i=p;i<=block_tot;i++)
- gcd_pre[i]=gcd(gcd_pre[i-],gcd_block[i]),
- xor_pre[i]=xor_pre[i-]^xor_block[i];
- v[L[p]].x=a[L[p]],v[L[p]].i=L[p];
- for (int i=L[p]+;i<=R[p];i++)
- v[i].x=v[i-].x^a[i],v[i].i=i;
- sort(v+L[p],v+R[p]+);
- }
- else
- {
- ll x=read();int ans=;
- for (int i=;i<=block_tot;i++)
- {
- if (ans) break;
- if (gcd_pre[i]==gcd_pre[i-])
- {
- if (x%gcd_pre[i-]==)
- {
- int l=L[i],r=R[i];
- while (l<=r)
- {
- int mid=l+r>>;
- if (v[mid].x>=(x/gcd_pre[i-]^xor_pre[i-])) ans=mid,r=mid-;
- else l=mid+;
- }
- if (ans&&v[ans].x==(x/gcd_pre[i-]^xor_pre[i-])) {ans=v[ans].i;break;}
- else ans=;
- }
- }
- else
- {
- int p=gcd_pre[i-],q=xor_pre[i-];
- for (int j=L[i];j<=R[i];j++)
- {
- p=gcd(p,a[j]),q^=a[j];
- if (x%p==&&x/p==q) {ans=j;break;}
- }
- }
- }
- if (ans) printf("%d\n",ans-);
- else printf("no\n");
- }
- }
- return ;
- }
BZOJ4028 HEOI2015公约数数列(分块)的更多相关文章
- [BZOJ4028][HEOI2015]公约数数列(分块)
先发掘性质: 1.xor和gcd均满足交换律与结合率. 2.前缀gcd最多只有O(log)个. 但并没有什么数据结构能同时利用这两个性质,结合Q=10000,考虑分块. 对每块记录这几个信息: 1.块 ...
- 【BZOJ4028】[HEOI2015]公约数数列 分块
[BZOJ4028][HEOI2015]公约数数列 Description 设计一个数据结构. 给定一个正整数数列 a_0, a_1, ..., a_{n - 1},你需要支持以下两种操作: 1. M ...
- BZOJ 4028: [HEOI2015]公约数数列 分块
4028: [HEOI2015]公约数数列 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4028 Description 设计一个数据结 ...
- bzoj4028: [HEOI2015]公约数数列
Description 设计一个数据结构. 给定一个正整数数列 a_0, a_1, ..., a_{n - 1},你需要支持以下两种操作: 1. MODIFY id x: 将 a_{id} 修改为 x ...
- [BZOJ4028][HAOI2015]公约数数列[分块+分析暴力]
题意 题目链接 分析 首先明确 \(xor\) 运算和 \(\rm gcd\) 没有联系! 注意到一个数字取 \(\rm gcd\) 且保证每次取 \(\rm gcd\) 值都会变小的话,最多取 \( ...
- 【BZOJ4028】[HEOI2015]公约数数列(分块)
[BZOJ4028][HEOI2015]公约数数列(分块) 题面 BZOJ 洛谷 题解 看一道题目就不会做系列 首先\(gcd\)最多只会有\(log\)种取值,所以我们可以暴力枚举出所有可能的\(g ...
- BZOJ 4028: [HEOI2015]公约数数列 【分块 + 前缀GCD】
任意门:https://www.lydsy.com/JudgeOnline/problem.php?id=4028 4028: [HEOI2015]公约数数列 Time Limit: 10 Sec ...
- 洛谷 P4108 / loj 2119 [HEOI2015] 公约数数列 题解【分块】
看样子分块题应该做的还不够. 题目描述 设计一个数据结构. 给定一个正整数数列 \(a_0, a_1, \ldots , a_{n-1}\),你需要支持以下两种操作: MODIFY id x: 将 \ ...
- luogu P4108 [HEOI2015]公约数数列——solution
-by luogu 不会啊.... 然后%了一发题解, 关键是 考虑序列{$a_n$}的前缀gcd序列, 它是单调不升的,且最多只会改变$log_2N$次,因为每变一次至少除2 于是,当我们询问x时: ...
随机推荐
- Mac下node.js安装与卸载
安装: 访问 http://nodejs.org/ 进入官网,下载 Mac 版本的 node.js,双击打开安装即可. 通过终端输入命令 node -v 验证 node 是否安装正确:npm -v 验 ...
- netty之粘包分包的处理
1.netty在进行字节数组传输的时候,会出现粘包和分包的情况.当个数据还好,如果数据量很大.并且不间断的发送给服务器,这个时候就会出现粘包和分包的情况. 2.简单来说:channelBuffer在接 ...
- selenium--特殊元素定位
该篇博客总结特殊元素(select.radio\checkbox.时间控件.文件上传.图片验证码.模拟鼠标操作.Js 或 JQuery调用)操作. 1.select @Test public void ...
- C 数据类型 常量 变量
一 数据类型 1. 什么是数据 生活中时时刻刻都在跟数据打交道 比如体重数据 血压数据 股价数据等 在我们使用计算机的过程中 会接触到各种各样的数据 有文档数据 图片数据 视频数据 还有聊QQ时产生的 ...
- 【if控制器】-(某种情况成立就执行的场景)
if 控制器 一般来判断某种特殊情况 成立,就执行. JEXL Expression to evaluate:此处直接填写需要进行判断的表达式即可 表达式支持: == 是否等于,如${__jex ...
- java学习笔记-8.对象的容纳
1.Iterator(迭代器)和Enumeration(枚举类),都是用来遍历集合的,他们都是接口.区别是Enumeration只能读取集合的数据,而Iterator可以对数据进行删除,Iterato ...
- 从零开始的Python学习Episode 6——字符串操作
字符串操作 一.输出重复字符串 print('smile'*6) #输出6个smile 二.通过引索输出部分字符串 print('smile'[1:]) print('smile'[1:3]) #输出 ...
- POJ 2987 Firing(最大流最小割の最大权闭合图)
Description You’ve finally got mad at “the world’s most stupid” employees of yours and decided to do ...
- 《javascript模式--by Stoyan Stefanov》书摘--函数
三.函数 1.函数的命名属性 // IE下不支持name属性 var foo = function bar () { // todo }; foo.name; // "bar" 2 ...
- 自测之Lesson7:设备文件操作
题目:请编写一个输入密码(不回显)的程序,要求通过设置终端来完成. 完成代码: #include <stdio.h> #include <unistd.h> #include ...