整除分块学习笔记+[CQOI2007]余数求和(洛谷P2261,BZOJ1257)
上模板题例题:
[CQOI2007]余数求和
题目大意:求 $\sum^n_{i=1}k\ mod\ i$ 的值。
等等……这题就学了三天C++的都会吧?
$1\leq n,k\leq 10^9$。(一口老血喷到屏幕上)
$O(n)$ 行不通了,考虑别的做法。
我们来看一下 $\lfloor\frac{x}{i}\rfloor$ 的值。
$x=9$:(不包括0,只有4种取值?)
i |
1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 |
x/i | 9 | 4 | 3 | 2 | 1 | 1 | 1 | 1 | 1 |
0 |
$x=12$:(不包括0,只有6种取值?)
i | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 |
x/i | 12 | 6 | 4 | 3 | 2 | 2 | 1 | 1 | 1 | 1 | 1 |
1 |
貌似 $\lfloor\frac{x}{i}\rfloor$ 取值数不是很多?
我们来估算一下 $\lfloor\frac{x}{i}\rfloor$ 的不同取值个数:
当 $1\leq i\leq \lfloor\sqrt{x}\rfloor$ 时,$i$ 都只有 $\lfloor\sqrt{x}\rfloor$ 个,不同的取值数肯定不会更多。
当 $\lfloor\sqrt{x}\rfloor\leq i\leq x$ 时,$1\leq\lfloor\frac{x}{i}\rfloor\leq\lfloor\sqrt{x}\rfloor$,不同的取值数肯定 $\leq\lfloor\sqrt{x}\rfloor$ 个。
综上,不同取值数是 $\sqrt{x}$ 级别的。
然后我们可以发现相同的数是连续的一段。那么我们可以通过这个特点把 $\lfloor\frac{x}{i}\rfloor$ 分成几段,每一段的数相等,那么这一段的和就是长度 $\times$ 这个相同的数。
因为不同取值只有 $\sqrt{x}$ 个,所以这样加速后的时间复杂度是 $O(\sqrt{x})$,比 $O(x)$ 快了不少。这就是整除分块。
回到原题。
求 $\sum^n_{i=1}k\ mod\ i$ 的值。
这个……看着和整除分块没什么大关系的样子?
我们看这个 $mod$ 真碍眼,把它拆开。
$k\ mod\ i=k-i\times\lfloor\frac{k}{i}\rfloor$
那么就有:
$\ \sum^n_{i=1}k\ mod\ i$
$=\sum^n_{i=1}k-i\times\lfloor\frac{k}{i}\rfloor$
$=nk-\sum^n_{i=1}i\times\lfloor\frac{k}{i}\rfloor$
后面这个式子貌似可以整除分块了……怎么算呢?
我们考虑 $[l,r]$ 这段区间的求和,其中 $\lfloor\frac{k}{i}\rfloor=x:i\in [l,r]$。
$\ \sum^r_{i=l}i\times\lfloor\frac{k}{i}\rfloor$
$=\sum^r_{i=l}i\times x$
$=x\sum^r_{i=l}i$
$=\frac{x(l+r)(r-l+1)}{2}$
这样就不是很难了。
话说讲了这么久也没讲怎么枚举一段相同区间的左端点和右端点。
我们这样扫描:
一开始 $l=1$ 显而易见。
求出对应的 $r$。
这个区间求完了,下一个 $l$ 应该是下一个还没扫过的位置,即 $l=r+1$。
一直重复直到 $l$ 到了上界,也就是扫完了。
怎么求对应的 $r$ 呢?
既然 $\lfloor\frac{k}{l}\rfloor=\lfloor\frac{k}{r}\rfloor$,且 $r$ 是右端点(最大)
那么 $r=\frac{k}{\frac{k}{l}}$。(当然可能要跟枚举上界取一个min,视情况而定)
整除分块模板大概如下:
for(int l=,r;l<=n;l=r+){
r=n/(n/l);
//do something...
}
那么这题代码实现就不难了。需要注意本题有不少坑点,详见代码。(没错,代码并没有你想象的那么长!)
时间复杂度貌似是 $O(\sqrt{min(k,n)})$,空间复杂度 $O(1)$
#include<iostream>
#include<cmath>
using namespace std;
typedef long long ll; //long long是需要的
ll n,k,ans;
int main(){
cin>>n>>k;
ans=n*k;
for(ll l=,r;l<=min(n,k);l=r+){ //与上界取min!
r=min(k/(k/l),n); //与上界取min!
ans-=(k/l)*(l+r)*(r-l+)/;
}
cout<<ans<<endl;
}
整除分块的超简短代码
另外再推荐几题。抱歉只找到一题,虽说也不错
整除分块学习笔记+[CQOI2007]余数求和(洛谷P2261,BZOJ1257)的更多相关文章
- 莫比乌斯反演&整除分块学习笔记
整除分块 用于计算$\sum_{i=1}^n f(\lfloor{n/i} \rfloor)*i$之类的函数 整除的话其实很多函数值是一样的,对于每一块一样的商集中处理即可 若一个商的左边界为l,则右 ...
- 莫比乌斯反演学习笔记+[POI2007]Zap(洛谷P3455,BZOJ1101)
先看一道例题:[POI2007]Zap BZOJ 洛谷 题目大意:$T$ 组数据,求 $\sum^n_{i=1}\sum^m_{j=1}[gcd(i,j)=k]$ $1\leq T\leq 50000 ...
- P2261 [CQOI2007]余数求和 【整除分块】
一.题面 P2261 [CQOI2007]余数求和 二.分析 参考文章:click here 对于整除分块,最重要的是弄清楚怎样求的分得的每个块的范围. 假设$ n = 10 ,k = 5 $ $$ ...
- [洛谷P2261] [CQOI2007]余数求和
洛谷题目链接:[CQOI2007]余数求和 题目背景 数学题,无背景 题目描述 给出正整数n和k,计算G(n, k)=k mod 1 + k mod 2 + k mod 3 + - + k mod n ...
- 洛谷 P2261 [CQOI2007]余数求和 解题报告
P2261 [CQOI2007]余数求和 题意: 求\(G(n,k)=\sum_{i=1}^n k \ mod \ i\) 数据范围: \(1 \le n,k \le 10^9\) \(G(n,k)\ ...
- 洛谷——P2261 [CQOI2007]余数求和
P2261 [CQOI2007]余数求和 关键在于化简公式,题目所求$\sum_{i=1}^{n}k\mod i$ 简化式子,也就是$\sum_{i=1}^{n}(k-\frac{k}{i}\time ...
- [Luogu 2261] CQOI2007 余数求和
[Luogu 2261] CQOI2007 余数求和 这一定是我迄今为止见过最短小精悍的省选题了,核心代码 \(4\) 行,总代码 \(12\) 行,堪比小凯的疑惑啊. 这题一看暴力很好打,然而 \( ...
- 题解 P2261【[CQOI2007]余数求和】
P2261[[CQOI2007]余数求和] 蒟蒻终于不看题解写出了一个很水的蓝题,然而题解不能交了 虽然还看了一下自己之前的博客 题目要求: \[\sum_{i=1}^{n}{k \bmod i} \ ...
- [Luogu P2261] [CQOI2007]余数求和 (取模计算)
题面 传送门:https://www.luogu.org/problemnew/show/P2261 Solution 这题显然有一个O(n)的直接计算法,60分到手. 接下来我们就可以拿出草稿纸推一 ...
随机推荐
- MB_SELECT_GR_BLOCKED_STOCK 读取物料收货冻结库存
MMBE 查询物料的当前库存,有一列是收货冻结库存(GR Blocked Stock),但是没有明细. 通过函数 MB_SELECT_GR_BLOCKED_STOCK 可以查询物料收货冻结库存的明细. ...
- 20155330 《网络攻防》 Exp4 恶意代码分析
20155330 <网络攻防> Exp4 恶意代码分析 实验后回答问题 (1)如果在工作中怀疑一台主机上有恶意代码,但只是猜想,所有想监控下系统一天天的到底在干些什么.请设计下你想监控的操 ...
- 20155333 《网络对抗》 Exp6 信息搜集与漏洞扫描
20155333 <网络对抗> Exp6 信息搜集与漏洞扫描 基础问题 哪些组织负责DNS,IP的管理? 全球根服务器均由美国政府授权的ICANN统一管理,负责全球的域名根服务器.DNS和 ...
- Eclipse中Svn插件配置
1. Svn插件配置教程 http://www.cnblogs.com/ruiati/p/3584120.html 2. Svn插件使用教程 http://wenku.baidu.com/link?u ...
- opencv配置(转)
1. 下载安装Opencv,去官网http://opencv.org/即可下载最新版本的Opencv,此处用的是Opencv 2.4.10 安装时傻瓜式的,最新版本的安装就是相当于解压到你指定的安装目 ...
- Django实现websocket完成实时通讯、聊天室、在线客服等
一 什么是Websocket WebSocket是一种在单个TCP连接上进行全双工通信的协议 WebSocket使得客户端和服务器之间的数据交换变得更加简单,允许服务端主动向客户端推送数据.在WebS ...
- C#图片处理(转zhjzwl/archive)
基本原理: 获取每一个像素值,然后处理这些每一个像素值. 原始图片: ISINBAEVA ~~~~~~~~ 一. 底片效果 原理: GetPixel方法获得每一点像素的值, 然后再使用SetPix ...
- 前端菜鸟起飞之学会ps切图
由于之前只顾着追求效率,没有学习过PS,但其实这是前端开发人员需要学会的技能之一,曾经看过一个大佬的前端经验分享说他在招聘时遇到不会切图的会直接pass掉,可见前端开发人员学会切图是多么重要.通过观看 ...
- 关于GitHub上传没有记录(小绿块不显示的问题)
最近开始使用上github来上传保存自己在学习中所写过的代码,打算将自己每天的成果能有个保存,然后就利用上GitHub这么一个利器. 听说GitHub的那个绿块是用来记录每天的上传记录的,结果我将代码 ...
- spring boot 2.0 源码分析(五)
在上一篇文章中我们详细分析了spring boot是如何准备上下文环境的,今天我们来看一下run函数剩余的内容.还是先把run函数贴出来: /** * Run the Spring applicati ...