浅谈根号分治

2023-10-12 01:52
文章标签 浅谈 分治 根号

本文主要是介绍浅谈根号分治,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

根号分治

根号分治是一种优美的暴力。

顾名思义,根号分治就是对于一个长度为 N N N的数列,将查询和修改分为 ≤ N \leq N N > N >N >N的两个部分来处理。将两个部分分别处理并拼在一起,来优化时间复杂度。


例题

洛谷P3396 哈希冲突

有一个长度为 n n n的序列 a a a m m m次操作,每次操作有两个类型:

  • 查询所有满足 k % x = y k\% x=y k%x=y a k a_k ak之和
  • a x a_x ax修改为 y y y

1 ≤ n ≤ 150000 , 1 ≤ m ≤ 150000 , 1 ≤ a i ≤ 1000 1\leq n\leq 150000,1\leq m\leq 150000,1\leq a_i\leq 1000 1n150000,1m150000,1ai1000

解题思路

首先,我们考虑暴力。

code

#include<bits/stdc++.h>
using namespace std;
int n,m,B,x,y,ans,a[150005];
char ch;
int main()
{scanf("%d%d",&n,&m);B=sqrt(n);for(int i=1;i<=n;i++){scanf("%d",&a[i]);}while(m--){ch=getchar();while(ch!='A'&&ch!='C') ch=getchar();scanf("%d%d",&x,&y);if(ch=='A'){ans=0;for(int i=y;i<=n;i+=x) ans+=a[i];printf("%d\n",ans);}else a[x]=y;}return 0;
}

这个暴力是 O ( n m ) O(nm) O(nm)的,我们考虑优化。

我们发现,如果当前是第一种操作且 x > n x>\sqrt n x>n 时,一次查询是 O ( n ) O(\sqrt n) O(n )的。然而,在 x ≤ n x\leq \sqrt n xn 时,时间复杂度就变大了。我们考虑对 x ≤ n x\leq \sqrt n xn 的部分特殊处理一下。

f i , j f_{i,j} fi,j表示模数为 i i i时所有下标模 i i i j j j的数之和,那么我们用 O ( n n ) O(n\sqrt n) O(nn )处理出 1 ≤ i ≤ s q r t n 1\leq i\leq sqrt n 1isqrtn 0 ≤ j < i 0\leq j<i 0j<i的所有 f i , j f_{i,j} fi,j之和。

那么,在查询时,若 x ≤ n x\leq \sqrt n xn ,直接在 f i , j f_{i,j} fi,j O ( 1 ) O(1) O(1)查询;若 x > n x>n x>n,则 O ( n ) O(\sqrt n) O(n )暴力查询。在修改时,我们 O ( n ) O(\sqrt n) O(n )修改 f i , j f_{i,j} fi,j,再 O ( 1 ) O(1) O(1)修改 a i a_i ai,那么时间复杂度就降为 O ( ( n + m ) n ) O((n+m)\sqrt n) O((n+m)n )

code

#include<bits/stdc++.h>
using namespace std;
int n,m,B,x,y,ans,a[150005],f[405][405];
char ch;
int main()
{scanf("%d%d",&n,&m);B=sqrt(n);for(int i=1;i<=n;i++){scanf("%d",&a[i]);}for(int i=1;i<=B;i++){for(int j=1;j<=n;j++) f[i][j%i]+=a[j];}while(m--){ch=getchar();while(ch!='A'&&ch!='C') ch=getchar();scanf("%d%d",&x,&y);if(ch=='A'){if(x<=B) printf("%d\n",f[x][y]);else{ans=0;for(int i=y;i<=n;i+=x) ans+=a[i];printf("%d\n",ans);}}else{for(int i=1;i<=B;i++) f[i][x%i]+=y-a[x];a[x]=y;}}return 0;
}

这篇关于浅谈根号分治的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/192410

相关文章

浅谈主机加固,六种有效的主机加固方法

在数字化时代,数据的价值不言而喻,但随之而来的安全威胁也日益严峻。从勒索病毒到内部泄露,企业的数据安全面临着前所未有的挑战。为了应对这些挑战,一种全新的主机加固解决方案应运而生。 MCK主机加固解决方案,采用先进的安全容器中间件技术,构建起一套内核级的纵深立体防护体系。这一体系突破了传统安全防护的局限,即使在管理员权限被恶意利用的情况下,也能确保服务器的安全稳定运行。 普适主机加固措施:

Codeforces Round #240 (Div. 2) E分治算法探究1

Codeforces Round #240 (Div. 2) E  http://codeforces.com/contest/415/problem/E 2^n个数,每次操作将其分成2^q份,对于每一份内部的数进行翻转(逆序),每次操作完后输出操作后新序列的逆序对数。 图一:  划分子问题。 图二: 分而治之,=>  合并 。 图三: 回溯:

浅谈PHP5中垃圾回收算法(Garbage Collection)的演化

前言 PHP是一门托管型语言,在PHP编程中程序员不需要手工处理内存资源的分配与释放(使用C编写PHP或Zend扩展除外),这就意味着PHP本身实现了垃圾回收机制(Garbage Collection)。现在如果去PHP官方网站(php.net)可以看到,目前PHP5的两个分支版本PHP5.2和PHP5.3是分别更新的,这是因为许多项目仍然使用5.2版本的PHP,而5.3版本对5.2并不是完

浅谈java向上转型和乡下转型

首先学习每一种知识都需要弄明白这知识是用来干什么使用的 简单理解:当对象被创建时,它可以被传递给这些方法中的任何一个,这意味着它依次被向上转型为每一个接口,由于java中这个设计接口的模式,使得这项工作不需要程序员付出任何特别的努力。 向上转型的作用:1、为了能够向上转型为多个基类型(由此而带来的灵活性) 2、使用接口的第二个原因却是与使用抽象基类相同,防止客户端创建该类的对象,并确保这仅仅

【前端安全】浅谈XSS攻击和防范

定义 XSS是跨站脚本攻击(Cross Site Scripting),为不和层叠样式表(Cascading Style Sheets, CSS)的缩写混淆,故将跨站脚本攻击缩写为XSS。 恶意攻击者往Web页面里插入恶意Script代码,当用户浏览该页之时,嵌入其中Web里面的Script代码会被执行,从而达到恶意攻击用户的目的。 分类 大分类小分类原理非存储DOM型① 不需要经过服务器

水处理过滤器运行特性及选择原则浅谈

过滤属于流体的净化过程中不可缺的处理环节,主要用于去除流体中的颗粒物或其他悬浮物。水处理过滤器的原理是利用有孔介质,从流体中去除污染物,使流体达到所需的洁净度水平。         水处理过滤器的滤壁是有一定厚度的,也就是说过滤器材具有深度,以“弯曲通 道”的形式对去除污染物起到了辅助作用。过滤器是除去液体中少量固体颗粒的设备,当流体进入置有一定规格滤网的滤筒后,其杂质被阻挡,而

浅谈NODE的NPM命令和合约测试开发工具HARDHAT

$ npm install yarn -g  # 将模块yarn全局安装 $ npm install moduleName # 安装模块到项目目录下 默认跟加参数 --save 一样 会在package文件的dependencies节点写入依赖。   $ npm install -g moduleName # -g 的意思是将模块安装到全局,具体安装到磁盘哪个位置,要看 npm root -g

浅谈SOC片上系统LoRa-STM32WLE5数据安全防御机制

随着物联网设备的普及,数以亿计的设备正在通过无线网络进行通信,传输大量的敏感数据。这种大规模的设备联网带来了便捷性,但也伴随着巨大的安全风险。SoC片上系统通过将无线通信、处理器、存储和安全机制集成在同一个芯片中,为物联网应用提供了高度集成的解决方案。这种设计大大简化了硬件开发流程,同时提高了设备的整体性能和安全性。SoC不仅能够满足长距离、低功耗的无线通信需求,还能通过先进的加密技术,确保数据在

浅谈RabbitMQ的基石—高级消息队列协议(AMQP)

点击上方蓝色字体,选择“设为星标” 回复”资源“获取更多资源 大数据技术与架构 点击右侧关注,大数据开发领域最强公众号! 大数据真好玩 点击右侧关注,大数据真好玩!     前言 自从去年做了不少流式系统(Flink也好,Spark Streaming也好)对接RabbitMQ的实时作业。之前一直都在Kafka的领域里摸爬滚打,对RabbitMQ只是有浅薄的了解而已。随着自己逐渐把R