原文链接https://www.cnblogs.com/zhouzhendong/p/51Nod1222.html

题意

  给定 $a,b$, 求

$$\sum_{n=a}^b \sum_{i=1}^n \sum_{j=1}^i [{\rm lcm } (i,j) = n]$$

$$a,b\leq 10^{11}$$

$${\rm Time \ Limit } = 6s$$

题解

  本题做法很多。

Min_25 筛

  先差分一下,转化成求前缀和。

  先把原题的统计无序数对转化成统计有序数对,最终 $ans' = (ans+n)/2$ 即可。

  设集合 $P$ 表示素数集合。

  设 $c(n,p)$ 表示最大的使得 $p^{c(n,p)}|n$ 的数。

  若 ${\rm lcm } (i,j) = n$ ,则

$$\forall p \in P, c(n,p)=\max(c(i,p),c(j,p))$$

  所以,$\forall p\in P$ ,$c(i,p)$ 和 $c(j,p)$ 共有 $2c(n,p) +1 $ 种取值方法。

  所以,设

$$n=\prod_i p_i^{k_i} (p_i\in P)$$

  则

$$ \sum_{i=1}^n \sum_{j=1}^i [{\rm lcm } (i,j) = n] = \prod_t (2k_t+1) $$

  显然这个式子满足 Min_25 筛的条件,直接筛就好了。

整除分块

$$S(n) = \sum_{i=1}^{n} \sigma_0(i^2)\\=\sum_{i=1}^n \sum_{d|i} 2^{\omega(d)}\\=\sum_{d=1}^n\lfloor \frac nd \rfloor 2^{\omega(d)}$$

整除分块一下,考虑

$$\sum_{i=1}^n 2^{\omega (i)} = \sum_{i=1}^n \sum_{d|i} \mu(d)^2\\=\sum_{d=1}^n\lfloor \frac nd  \rfloor \mu(d) ^2$$

再整除分块,再推

$$\sum_{i=1}^n \mu (i)^2 = \sum_{i=1}^n \mu(i) \lfloor \frac n {i^2} \rfloor$$

于是只要求个 $\mu$ 的前缀和。

看起来要杜教筛,但是……

由于这里 $i^2\leq n$,所以只需要暴力预处理前缀和就好了。

注意这个做法可能会被卡常数!

复杂度正确的暴力

$$S(k) = \sum_{n=1}^k\sum_{i=1}^{n}\sum_{j=1}^n [{\rm lcm}(i,j) = n]$$

$$=\sum_{n=1}^k\sum_{d=1}^n\sum_{i=1}^n\sum_{j=1}^n[ijd=n] [\gcd(i,j) = 1]$$

$$=\sum_{p=1}^k\mu(p) \sum_{i,j,d} [ijd\leq \frac{k} {p^2}]$$

先枚举个 $p$ ,再强制 $i,j,d$ 单调不降,枚举 $i$ 再枚举 $j$ ,暴力计算。

证明一下时间复杂度:

$$T(n) = O\left(\sum_{p=1}^{\sqrt n} \sum_{i=1}^{\sqrt[3]{\frac{n}{p^2}}} \sqrt{\left\lfloor \frac{n}{p^2i}\right\rfloor}\right) $$

$$\because O\left(\sum_{i=1}^{\sqrt[3]{n}}\sqrt{ \frac n i}\right) = O\left(\int _{1}^{\sqrt[3]{n}} \sqrt{\frac n x} {\rm d} x\right)\\=O\left(\sqrt{n}\int _{1}^{\sqrt[3]{n}} x^{-0.5} {\rm d} x\right)=O\left(\sqrt{n}\cdot 0.5 \left(\sqrt[3]{n}\right) ^{0.5}\right)=O\left(n^{\frac 2 3 }\right)$$

$$\therefore T(n) = O\left(\sum_{p=1}^{\sqrt n} \sum_{i=1}^{\sqrt[3]{\frac{n}{p^2}}} \sqrt{\lfloor \frac{n}{p^2i}\rfloor}\right)=O\left(\sum_{p=1}^{\sqrt{n}} \left(\frac n {p^2} \right) ^{\frac 2 3 } \right)=O\left(n^{\frac 2 3}\int _1^\sqrt n x^{-\frac 4 3} {\rm d} x \right) \\=O\left( n^\frac 23 \cdot -1\cdot (\left (\sqrt n\right)^{-\frac 13} - 1)\right) = O\left(n^\frac 23 (1-n^{-\frac 16})\right) = O\left(n^\frac 23\right)$$

没想到这个“暴力”的复杂度居然是对的!代码短小精悍,吊打前几种做法了!

代码1 - Min_25 筛

#pragma GCC optimize("Ofast","inline")
#include <bits/stdc++.h>
#define clr(x) memset(x,0,sizeof (x))
using namespace std;
typedef long long LL;
LL read(){
LL x=0,f=0;
char ch=getchar();
while (!isdigit(ch))
f|=ch=='-',ch=getchar();
while (isdigit(ch))
x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return f?-x:x;
}
const int Base=1000005,N=Base*2+5;
LL n,cn,a,b,base;
LL h[N],ps[N],cnt;
LL p[N],pcnt;
#define ID(i) ((i)<=base?i:cnt-cn/(i)+1)
LL f(int e){
return e*2+1;
}
LL g(LL n,LL m){
LL ans=max(0LL,h[ID(n)]-h[ID(p[m-1])]);
for (int i=m;i<=pcnt&&p[i]*p[i]<=n;i++){
LL nn=n/p[i];
for (int e=1;nn>0;e++,nn/=p[i])
ans+=f(e)*((e>1)+g(nn,i+1));
}
return ans;
}
LL _solve(LL _n){
cn=n=_n,base=(LL)sqrt(n),cnt=pcnt=0;
for (LL i=1;i<=n;i=ps[cnt]+1)
ps[++cnt]=n/(n/i),h[cnt]=ps[cnt]-1;
p[0]=1;
for (LL i=2;i<=base;i++)
if (h[i]!=h[i-1]){
p[++pcnt]=i;
LL i2=i*i;
for (LL j=cnt;ps[j]>=i2;j--)
h[j]-=h[ID(ps[j]/i)]-(pcnt-1);
}
for (LL i=1;i<=cnt;i++)
h[i]*=3;
return g(n,1)+1;
}
LL solve(LL n){
return (_solve(n)+n)/2;
}
int main(){
a=read(),b=read();
cout<<solve(b)-solve(a-1)<<endl;
return 0;
}

代码2 - 整除分块

#include <bits/stdc++.h>
#define clr(x) memset(x,0,sizeof (x))
using namespace std;
typedef long long LL;
LL read(){
LL x=0,f=0;
char ch=getchar();
while (!isdigit(ch))
f|=ch=='-',ch=getchar();
while (isdigit(ch))
x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return f?-x:x;
}
const int N=11000005;
const LL nb=1e11,un=N-5;
int p[N/5],pcnt;
bool vis[N],_u2[N];
char u[N];
int u2[N],w[N];
unordered_map <LL,LL> U2,W;
LL A;
void get_prime(int n){
pcnt=0,u[1]=1;
for (register int i=2;i<=n;i++){
if (!vis[i])
p[++pcnt]=i,u[i]=-1,w[i]=1;
for (register int j=1;j<=pcnt&&p[j]*i<=n;j++){
vis[i*p[j]]=1;
if (i%p[j])
u[i*p[j]]=-u[i],w[i*p[j]]=w[i]+1;
else {
u[i*p[j]]=0,w[i*p[j]]=w[i];
break;
}
}
}
}
void prework(int n){
get_prime(n);
for (int i=1;i<=n;i++){
_u2[i]=u[i]*u[i];
// u[i]+=u[i-1];
u2[i]=u2[i-1]+_u2[i];
w[i]=1LL<<w[i];
w[i]+=w[i-1];
}
}
LL getU2(LL n){
if (n<=un)
return u2[n];
if (U2.count(n))
return U2[n];
LL ans=0;
ans=0;
for (LL i=1;i*i<=n;i++)
ans+=u[i]?u[i]*(n/(i*i)):0;
return U2[n]=ans;
}
LL getW(LL n){
if (n<=un)
return w[n];
if (W.count(n))
return W[n];
LL ans=0;
LL pr=0,nw,sq=(LL)sqrt(n);
for (LL i=1;i<=sq;i++)
ans+=_u2[i]?n/i:0;
pr=u2[sq];
for (LL i=sq+1,j;i<=n;i=j+1){
j=n/(n/i);
nw=j<=un?u2[j]:getU2(j);
ans+=(nw-pr)*(n/i);
pr=nw;
}
return W[n]=ans;
}
LL getS(LL n){
if (!n)
return 0;
LL ans=0,pr=0,nw;
for (LL i=1,j;i<=n;i=j+1){
j=n/(n/i);
nw=j<=un?w[j]:getW(j);
ans+=(nw-pr)*(n/i);
pr=nw;
}
return ans;
}
LL solve(LL n){
if (!n)
return 0;
A=n;
return (getS(n)+n)/2;
}
int main(){
prework(N-5);
LL a=read(),b=read();
cout<<solve(b)-solve(a-1)<<endl;
return 0;
}

代码3 - 优秀的暴力

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int N=5e5+10;
int mu[N],l,r;
int calc(int x){
int re=0;
for (int i=1;i*i<=x;i++)
if (mu[i]){
int tmp=0,m=x/(i*i);
for (int j=1;j*j*j<=m;j++){
tmp+=(m/(j*j)-j)*3+1;
for (int k=j+1;j*k*k<=m;k++)
tmp+=(m/(j*k)-k)*6+3;
}
re+=mu[i]*tmp;
}
return (re+x)/2;
}
signed main(){
mu[1]=1;
for (int i=1;i<N;i++)
for (int j=i+i;j<N;j+=i)
mu[j]-=mu[i];
scanf("%lld%lld",&l,&r);
printf("%lld",calc(r)-calc(l-1));
return 0;
}

  

51Nod1222 最小公倍数计数 数论 Min_25 筛的更多相关文章

  1. LOJ3069. 「2019 集训队互测 Day 1」整点计数(min_25筛)

    题目链接 https://loj.ac/problem/3069 题解 复数真神奇. 一句话题意:令 \(f(x)\) 表示以原点 \((0, 0)\) 为圆心,半径为 \(x\) 的圆上的整点数量, ...

  2. 模板 - 数学 - 数论 - Min_25筛

    终于知道发明者的正确的名字了,是Min_25,这个筛法速度为亚线性的\(O(\frac{n^{\frac{3}{4}}}{\log x})\),用于求解具有下面性质的积性函数的前缀和: 在 \(p\) ...

  3. 51nod1222 最小公倍数计数

    题目来源: Project Euler 基准时间限制:6 秒 空间限制:131072 KB 分值: 640  定义F(n)表示最小公倍数为n的二元组的数量. 即:如果存在两个数(二元组)X,Y(X & ...

  4. 51nod1222最小公倍数计数

    51nod1222 http://210.33.19.103/contest/1113/problem/2 同学的神仙做法: 首先考虑先去掉X<=Y的限制,也就是先计算满足要求的任意有序pair ...

  5. 51nod1222 最小公倍数计数 莫比乌斯反演 数学

    求$\sum_{i = 1}^{n} \sum_{j = 1}^{i} [lcm(i, j) \le n]$因为这样不好求,我们改成求$\sum_{i = 1}^{n} \sum_{j = 1}^{n ...

  6. [51nod1222] 最小公倍数计数(莫比乌斯反演)

    题面 传送门 题解 我此生可能注定要和反演过不去了--死都看不出来为啥它会突然繁衍反演起来啊-- 设\(f(n)=\sum_{i=1}^n\sum_{j=1}^n[{ij\over\gcd(i,j)} ...

  7. Min_25 筛 学习笔记

    原文链接https://www.cnblogs.com/zhouzhendong/p/Min-25.html 前置技能 埃氏筛法 整除分块(这里有提到) 本文概要 1. 问题模型 2. Min_25 ...

  8. 【51nod】1222 最小公倍数计数 莫比乌斯反演+组合计数

    [题意]给定a和b,求满足a<=lcm(x,y)<=b && x<y的数对(x,y)个数.a,b<=10^11. [算法]莫比乌斯反演+组合计数 [题解]★具体 ...

  9. 51NOD 1222 最小公倍数计数 [莫比乌斯反演 杜教筛]

    1222 最小公倍数计数 题意:求有多少数对\((a,b):a<b\)满足\(lcm(a,b) \in [1, n]\) \(n \le 10^{11}\) 卡内存! 枚举\(gcd, \fra ...

随机推荐

  1. 利用fis3构建前端项目工程

    FIS3是国内百度公司产出的一款前端工程构建工具,FIS3可以解决前端工程中性能优化.资源加载(异步.同步.按需.预加载.依赖管理.合并.内嵌).模块化开发.自动化工具.开发规范.代码部署等问题,首先 ...

  2. min-max容斥学习笔记

    min-max容斥学习笔记 前置知识 二项式反演 \[ f(n)=\sum_{i=0}^n\binom{n}{i}g(i)\Leftrightarrow g(n)=\sum_{i=0}^n(-1)^{ ...

  3. [NOI2005]维护数列——平衡树观止

    本题题解并不详细,不推荐大家看这一篇. 可以看这篇 题目描述 请写一个程序,要求维护一个数列,支持以下 6 种操作:(请注意,格式栏 中的下划线‘ _ ’表示实际输入文件中的空格) 100%的数据中, ...

  4. Redux Counter Vanilla example

    此示例不需要构建系统或视图框架,并且存在以显示与ES5一起使用的原始Redux API. 代码如下 <!DOCTYPE html> <html> <head> &l ...

  5. CentOS Linux安装python3

    本文的方法是在CentOS上新装了python3,如果本机安装了python2则保留,因为可能有程序依赖目前的python2环境,比如yum!!!!! 一.安装python3.7 1. 安装依赖环境 ...

  6. 强化学习Q-Learning算法详解

    python风控评分卡建模和风控常识(博客主亲自录制视频教程) https://study.163.com/course/introduction.htm?courseId=1005214003&am ...

  7. Ubuntu16下Hive 安装

    0.安装环境和版本 Ubuntu16,hadoop版本是2.7.2 ,选择Hive版本为  hive-2.1.17 1. Hive安装包下载 地址: https://mirrors.tuna.tsin ...

  8. win7 远程连接服务器出现身份验证错误,又找不到加密Oracle修正

    今天想用远程桌面连接登录服务器,结果,弹出一个错误的提示框:发生身份验证错误,要求的函数不受支持. 然后在网上找了相关的教程,基本上所有的方法都是如下所示: 策略路径:"计算机配置" ...

  9. oldboy s21day07(深浅拷贝及文件操作)

    #!/usr/bin/env python# -*- coding:utf-8 -*- # 1.看代码写结果'''v1 = [1, 2, 3, 4, 5]v2 = [v1, v1, v1]v1.app ...

  10. CSS margin负值学习及实际应用

    前言 margin属性在实际中非常常用,也是平时踩坑较多的地方.margin折叠部分相信不少人都因为这样那样的原因中过招.margin负值也是很常用的功能,很多特殊的布局方法都依赖于它. 表现 虽然m ...