根号分治

前言

本题是一道讲解根号分治思想的论文题(然鹅我并没有找到论文),正

如论文中所说,根号算法——不仅是分块,根号分治利用的思想和分块像

似却又不同,某一篇洛谷日报中说过,分块算法实质上是一种是通过分成

多块后在每块上打标记以实现快速区间修改,区间查询的一种算法。根号

分治与其思路相似,将原本若一次性解决时间复杂度很高的问题分块去解

决来降低整体的时间复杂度。

例题

以本题举例子哈希冲突

本题作为论文的第一道题目,是一道很好的练习题,注意,本体给出的

\(value[i]\) 是 \(i\) 在序列中出现的次数,不要把题读错了(一开始我就读错了)

我们首先阅读题目,发现,无论是 \(O(n^2)\) 预处理, \(O(1)\) 查询,还是在

查询时直接\(O(n^2)\)获取答案,都是 \(O(n^2)\) 的时间复杂度,我们考虑对

其进行优化,我们可以考虑将\(p\),也就是模数按根号分块处理

对于 \(p<=\sqrt{q}\) 我们直接 \(O(n\sqrt{n})\) 进行预处理,

  1. for(int i=1;i<=n;i++){
  2. v=read();
  3. val[i]=v;
  4. }
  5. size=sqrt(n);
  6. for(int i=1;i<=n;i++){
  7. for(int p=1;p<=size;p++){
  8. ans[p][i%p]+=val[i];
  9. }
  10. }

\(ans[p][i]\) 表示在 \(%p\) 后值为 \(i\)的数的个数

对 \(p>\sqrt{q}\) 的情况,

我们直接暴力得出答案,暴力得到答案的时间复杂度为 \(O(\sqrt{n})\)

  1. int an=val[y];
  2. for(int i=x+y;i<=n;i+=x){
  3. an+=val[i];
  4. }
  5. cout<<an<<endl;

\(an=val[y]\) 是为了处理 \(y%x=y\) \((y<x)\) 的情况,为什么说我们暴力跳

答案时间复杂度是 \(O(\sqrt{n})\) 呢?我们当前的 \(p>\sqrt{n})\) 因为统

计答案时每一次要跳\(p\)个数,所以有贡献的数一定小于 \(\frac{n}
{p}\) 也就是 \(\sqrt{n}\) ,所以暴力得到答案的时间复杂度为 \(O(\sqrt{n})\)

每一次修改,我们只需要修改 \(p<=\sqrt{n}\) 的情况即可,时间复杂度也是 \(O(\sqrt{n})\)

  1. cin>>x>>y;
  2. int l=y-val[x];
  3. val[x]=y;
  4. for(int p=1;p<=size;p++){
  5. ans[p][x%p]+=l;
  6. }

所以在本题,我们运用根号分治的想法,把时间复杂度由原本的\(O(n^2)\)

优化到了 \(O(n\sqrt{n})\) 从而解决本题。

莫名觉得根号分治挺像折半搜索

,推荐一道练习题CF444D DZY Loves Strings

还是很有难度的

代码

放一下全部代码吧

  1. #include<iostream>
  2. #include<string>
  3. #include<string>
  4. #include<cstdio>
  5. #include<cmath>
  6. #define int long long
  7. using namespace std;
  8. const int maxn=3e5+10;
  9. inline int read(){
  10. int ret=0;
  11. int f=1;
  12. char ch=getchar();
  13. while(ch<'0'||ch>'9'){
  14. if(ch=='-')
  15. f=-f;
  16. ch=getchar();
  17. }
  18. while(ch<='9'&&ch>='0'){
  19. ret=ret*10+(ch^'0');
  20. ch=getchar();
  21. }
  22. return ret*f;
  23. }
  24. int val[maxn];
  25. int n,m;
  26. int p;
  27. int ans[2000][2000];
  28. int size;
  29. char a;
  30. signed main(){
  31. // freopen("a.in","r",stdin);
  32. n=read();
  33. m=read();
  34. int v;
  35. for(int i=1;i<=n;i++){
  36. v=read();
  37. val[i]=v;//????????
  38. }
  39. size=sqrt(n);//????
  40. for(int i=1;i<=n;i++){
  41. for(int p=1;p<=size;p++){
  42. ans[p][i%p]+=val[i];
  43. }
  44. }
  45. int x,y;
  46. while(m--){
  47. cin>>a;
  48. if(a=='A'){
  49. x=read();
  50. y=read();
  51. if(x<=size){
  52. cout<<ans[x][y]<<endl;
  53. }
  54. else{
  55. int an=val[y];
  56. for(int i=x+y;i<=n;i+=x){
  57. an+=val[i];
  58. }
  59. cout<<an<<endl;
  60. }
  61. }
  62. if(a=='C'){
  63. cin>>x>>y;
  64. int l=y-val[x];
  65. val[x]=y;
  66. for(int p=1;p<=size;p++){
  67. ans[p][x%p]+=l;
  68. }
  69. }
  70. }
  71. return 0;
  72. }

到这里本题解就结束了

完结撒花!!

题解 洛谷 P3396 【哈希冲突】(根号分治)的更多相关文章

  1. 洛谷P3396 哈希冲突 (分块)

    洛谷P3396 哈希冲突 题目背景 此题约为NOIP提高组Day2T2难度. 题目描述 众所周知,模数的hash会产生冲突.例如,如果模的数p=7,那么4和11便冲突了. B君对hash冲突很感兴趣. ...

  2. 洛谷 P3396 哈希冲突 解题报告

    P3396 哈希冲突 题目背景 此题约为NOIP提高组Day2T2难度. 题目描述 众所周知,模数的hash会产生冲突.例如,如果模的数p=7,那么4和11便冲突了. B君对hash冲突很感兴趣.他会 ...

  3. 洛谷P3396 哈希冲突(分块)

    传送门 题解在此,讲的蛮清楚的->这里 我就贴个代码 //minamoto #include<iostream> #include<cstdio> #include< ...

  4. 洛谷P3396 哈希冲突

    分块还真是应用广泛啊...... 题意:求 解:以n0.5为界. 当p小于n0.5的时候,直接用p²大小的数组储存答案. 预处理n1.5,修改n0.5. 当p大于n0.5的时候,直接按照定义计算,复杂 ...

  5. 洛谷P3396哈希冲突

    传送门啦 非常神奇的分块大法. 这个题一看数据范围,觉得不小,但是如果我们以 $ \sqrt(x) $ 为界限,数据范围就降到了 $ x < 400 $ 我们设数组 $ f[i][j] $ 表示 ...

  6. P 3396 哈希冲突 根号分治

    Link 据说这是一道论文题????.具体论文好像是 集训队论文<根号算法--不只是分块> 根号分治的裸题. 首先我们考虑暴力怎么打. 先预处理出每个模数的答案,之后再 O(1) 的回答, ...

  7. P3396 哈希冲突(思维+方块)

    题目 P3396 哈希冲突 做法 预处理模数\([1,\sqrt{n}]\)的内存池,\(O(n\sqrt{n})\) 查询模数在范围里则直接输出,否则模拟\(O(m\sqrt{n})\) 修改则遍历 ...

  8. 洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP

    洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP 题目描述 佳媛姐姐过生日的时候,她的小伙伴从某宝上买了一个有趣的玩具送给他. 玩具上有一个数列,数列中某些项的值可能会 ...

  9. 题解 洛谷P5018【对称二叉树】(noip2018T4)

    \(noip2018\) \(T4\)题解 其实呢,我是觉得这题比\(T3\)水到不知道哪里去了 毕竟我比较菜,不大会\(dp\) 好了开始讲正事 这题其实考察的其实就是选手对D(大)F(法)S(师) ...

随机推荐

  1. nginx下配置php5和php7

    用的是lnmp 一键安装的 php5.6版本网上百度Ubuntu安装多版本PHP就行 参考文章原链接:http://blog.csdn.net/21aspnet/article/details/476 ...

  2. 【译】关于Rust模块的清晰解释

    原文链接: http://www.sheshbabu.com/posts/rust-module-system/ 原文标题: Clear explanation of Rust's module sy ...

  3. 安装使用Pycharm及Anaconda最全教程

    网上安装anaconda和pycharm的教程很多,然而很少有人能够很详细地讲解,特别是对于pycharm的虚拟环境相关的说明很少,我也是懵逼的用了两年多,经常发现之前pycharm安装的第三方库,明 ...

  4. Spider--补充--selenium的使用

    # Selenium (firefox) # 1,介绍: # selenium 是一个 web 的自动化测试工具,是一个包,可以支持 C. java.ruby.python.或都是 C# 语言. # ...

  5. pandas_知识总结_基础

    # Pandas 知识点总结 # Pandas数据结构:Series 和 DataFrame import pandas as pd import numpy as np # 一,Series: # ...

  6. jenkins配置邮件报错:501 mail from address must be same as authorization user

    jenkins配置文件的时候,遇到如下报错: 我的配置是这样的: 最后发现是jenkins url下面的系统管理员邮件地址没写,填写与用户名一致就可以了.

  7. 年轻人不讲武德来白piao我这个老同志

    朋友们好啊,我是码农小胖哥. 今天有个同学问我在吗,我说什么事? 给我发个截图,我一看!噢,原来是帮忙搞个定时任务,还是动态的. 他说了两种选择,一种是用DelayQueue,一种是用消息队列. 他说 ...

  8. 机器学习——dbscan密度聚类

    完整版可关注公众号:大数据技术宅获取 DBSCAN(Density-Based Spatial Clustering of Applications with Noise,基于密度的有噪应用中的空间聚 ...

  9. 自动化翻译ceph文档

    需求很简单,翻译官网的操作文档 下载ceph代码luminous版本 这个只用来编译doc的,我们只需要最新的这个分支即可,拉最少的代码 git clone -b v12.2.13 --single- ...

  10. 利用火焰图分析ceph pg分布

    前言 性能优化大神Brendan Gregg发明了火焰图来定位性能问题,通过图表就可以发现问题出在哪里,通过svg矢量图来查看性能卡在哪个点,哪个操作占用的资源最多 在查看了原始数据后,这个分析的原理 ...