前缀gcd的变化次数是log的,考虑对每一种gcd查询,问题变为查询一段区间是否存在异或前缀和=x/gcd。

  无修改的话显然可以可持久化trie,但这玩意实在没法支持修改。于是考虑分块。

  对于每一块将其中所有块内异或前缀和排序。查询时先看这块与上一块相比gcd有没有变化,如果有对其中每个位置暴力查询,否则在排序后的数组中二分。修改时暴力改每一块的前缀gcd及异或和,被修改的块暴力重构排序数组即可。

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cmath>
  4. #include<cstdlib>
  5. #include<cstring>
  6. #include<algorithm>
  7. using namespace std;
  8. #define N 100010
  9. #define ll long long
  10. ll read()
  11. {
  12. ll x=,f=;char c=getchar();
  13. while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
  14. while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
  15. return x*f;
  16. }
  17. int n,m,a[N];
  18. int block_size,block_tot,L[N],R[N],pos[N];
  19. int gcd_pre[N],xor_pre[N],gcd_block[N],xor_block[N];
  20. struct data
  21. {
  22. int x,i;
  23. bool operator <(const data&a) const
  24. {
  25. return x<a.x||x==a.x&&i<a.i;
  26. }
  27. }v[N];
  28. int gcd(int n,int m){return m==?n:gcd(m,n%m);}
  29. int main()
  30. {
  31. #ifndef ONLINE_JUDGE
  32. freopen("bzoj4028.in","r",stdin);
  33. freopen("bzoj4028.out","w",stdout);
  34. const char LL[]="%I64d\n";
  35. #else
  36. const char LL[]="%lld\n";
  37. #endif
  38. n=read();
  39. for (int i=;i<=n;i++) a[i]=read();
  40. block_size=sqrt(n);block_tot=(n-)/block_size+;
  41. for (int i=;i<=block_tot;i++)
  42. L[i]=(i-)*block_size+,R[i]=min(n,i*block_size);
  43. for (int i=;i<=block_tot;i++)
  44. {
  45. for (int j=L[i];j<=R[i];j++)
  46. gcd_block[i]=gcd(gcd_block[i],a[j]),
  47. xor_block[i]=xor_block[i]^a[j],
  48. pos[j]=i;
  49. gcd_pre[i]=gcd(gcd_pre[i-],gcd_block[i]),xor_pre[i]=xor_pre[i-]^xor_block[i];
  50. }
  51. for (int i=;i<=block_tot;i++)
  52. {
  53. v[L[i]].x=a[L[i]],v[L[i]].i=L[i];
  54. for (int j=L[i]+;j<=R[i];j++)
  55. v[j].x=v[j-].x^a[j],v[j].i=j;
  56. sort(v+L[i],v+R[i]+);
  57. }
  58. m=read();
  59. while (m--)
  60. {
  61. char c=getchar();
  62. while (c<'A'||c>'Z') c=getchar();
  63. if (c=='M')
  64. {
  65. int p=read()+,x=read();a[p]=x;p=pos[p];
  66. gcd_block[p]=xor_block[p]=;
  67. for (int i=L[p];i<=R[p];i++)
  68. gcd_block[p]=gcd(gcd_block[p],a[i]),
  69. xor_block[p]=xor_block[p]^a[i];
  70. for (int i=p;i<=block_tot;i++)
  71. gcd_pre[i]=gcd(gcd_pre[i-],gcd_block[i]),
  72. xor_pre[i]=xor_pre[i-]^xor_block[i];
  73. v[L[p]].x=a[L[p]],v[L[p]].i=L[p];
  74. for (int i=L[p]+;i<=R[p];i++)
  75. v[i].x=v[i-].x^a[i],v[i].i=i;
  76. sort(v+L[p],v+R[p]+);
  77. }
  78. else
  79. {
  80. ll x=read();int ans=;
  81. for (int i=;i<=block_tot;i++)
  82. {
  83. if (ans) break;
  84. if (gcd_pre[i]==gcd_pre[i-])
  85. {
  86. if (x%gcd_pre[i-]==)
  87. {
  88. int l=L[i],r=R[i];
  89. while (l<=r)
  90. {
  91. int mid=l+r>>;
  92. if (v[mid].x>=(x/gcd_pre[i-]^xor_pre[i-])) ans=mid,r=mid-;
  93. else l=mid+;
  94. }
  95. if (ans&&v[ans].x==(x/gcd_pre[i-]^xor_pre[i-])) {ans=v[ans].i;break;}
  96. else ans=;
  97. }
  98. }
  99. else
  100. {
  101. int p=gcd_pre[i-],q=xor_pre[i-];
  102. for (int j=L[i];j<=R[i];j++)
  103. {
  104. p=gcd(p,a[j]),q^=a[j];
  105. if (x%p==&&x/p==q) {ans=j;break;}
  106. }
  107. }
  108. }
  109. if (ans) printf("%d\n",ans-);
  110. else printf("no\n");
  111. }
  112. }
  113. return ;
  114. }

BZOJ4028 HEOI2015公约数数列(分块)的更多相关文章

  1. [BZOJ4028][HEOI2015]公约数数列(分块)

    先发掘性质: 1.xor和gcd均满足交换律与结合率. 2.前缀gcd最多只有O(log)个. 但并没有什么数据结构能同时利用这两个性质,结合Q=10000,考虑分块. 对每块记录这几个信息: 1.块 ...

  2. 【BZOJ4028】[HEOI2015]公约数数列 分块

    [BZOJ4028][HEOI2015]公约数数列 Description 设计一个数据结构. 给定一个正整数数列 a_0, a_1, ..., a_{n - 1},你需要支持以下两种操作: 1. M ...

  3. BZOJ 4028: [HEOI2015]公约数数列 分块

    4028: [HEOI2015]公约数数列 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4028 Description 设计一个数据结 ...

  4. bzoj4028: [HEOI2015]公约数数列

    Description 设计一个数据结构. 给定一个正整数数列 a_0, a_1, ..., a_{n - 1},你需要支持以下两种操作: 1. MODIFY id x: 将 a_{id} 修改为 x ...

  5. [BZOJ4028][HAOI2015]公约数数列[分块+分析暴力]

    题意 题目链接 分析 首先明确 \(xor\) 运算和 \(\rm gcd\) 没有联系! 注意到一个数字取 \(\rm gcd\) 且保证每次取 \(\rm gcd\) 值都会变小的话,最多取 \( ...

  6. 【BZOJ4028】[HEOI2015]公约数数列(分块)

    [BZOJ4028][HEOI2015]公约数数列(分块) 题面 BZOJ 洛谷 题解 看一道题目就不会做系列 首先\(gcd\)最多只会有\(log\)种取值,所以我们可以暴力枚举出所有可能的\(g ...

  7. BZOJ 4028: [HEOI2015]公约数数列 【分块 + 前缀GCD】

    任意门:https://www.lydsy.com/JudgeOnline/problem.php?id=4028 4028: [HEOI2015]公约数数列 Time Limit: 10 Sec   ...

  8. 洛谷 P4108 / loj 2119 [HEOI2015] 公约数数列 题解【分块】

    看样子分块题应该做的还不够. 题目描述 设计一个数据结构. 给定一个正整数数列 \(a_0, a_1, \ldots , a_{n-1}\),你需要支持以下两种操作: MODIFY id x: 将 \ ...

  9. luogu P4108 [HEOI2015]公约数数列——solution

    -by luogu 不会啊.... 然后%了一发题解, 关键是 考虑序列{$a_n$}的前缀gcd序列, 它是单调不升的,且最多只会改变$log_2N$次,因为每变一次至少除2 于是,当我们询问x时: ...

随机推荐

  1. Mac下node.js安装与卸载

    安装: 访问 http://nodejs.org/ 进入官网,下载 Mac 版本的 node.js,双击打开安装即可. 通过终端输入命令 node -v 验证 node 是否安装正确:npm -v 验 ...

  2. netty之粘包分包的处理

    1.netty在进行字节数组传输的时候,会出现粘包和分包的情况.当个数据还好,如果数据量很大.并且不间断的发送给服务器,这个时候就会出现粘包和分包的情况. 2.简单来说:channelBuffer在接 ...

  3. selenium--特殊元素定位

    该篇博客总结特殊元素(select.radio\checkbox.时间控件.文件上传.图片验证码.模拟鼠标操作.Js 或 JQuery调用)操作. 1.select @Test public void ...

  4. C 数据类型 常量 变量

    一 数据类型 1. 什么是数据 生活中时时刻刻都在跟数据打交道 比如体重数据 血压数据 股价数据等 在我们使用计算机的过程中 会接触到各种各样的数据 有文档数据 图片数据 视频数据 还有聊QQ时产生的 ...

  5. 【if控制器】-(某种情况成立就执行的场景)

    if 控制器   一般来判断某种特殊情况 成立,就执行. JEXL Expression to evaluate:此处直接填写需要进行判断的表达式即可 表达式支持: ==  是否等于,如${__jex ...

  6. java学习笔记-8.对象的容纳

    1.Iterator(迭代器)和Enumeration(枚举类),都是用来遍历集合的,他们都是接口.区别是Enumeration只能读取集合的数据,而Iterator可以对数据进行删除,Iterato ...

  7. 从零开始的Python学习Episode 6——字符串操作

    字符串操作 一.输出重复字符串 print('smile'*6) #输出6个smile 二.通过引索输出部分字符串 print('smile'[1:]) print('smile'[1:3]) #输出 ...

  8. POJ 2987 Firing(最大流最小割の最大权闭合图)

    Description You’ve finally got mad at “the world’s most stupid” employees of yours and decided to do ...

  9. 《javascript模式--by Stoyan Stefanov》书摘--函数

    三.函数 1.函数的命名属性 // IE下不支持name属性 var foo = function bar () { // todo }; foo.name; // "bar" 2 ...

  10. 自测之Lesson7:设备文件操作

    题目:请编写一个输入密码(不回显)的程序,要求通过设置终端来完成. 完成代码: #include <stdio.h> #include <unistd.h> #include ...