数论函数及其求和 待更新

主要记录一下相关的题目,略过知识讲解。

例题1 数论分块

P3935 Calculating

题目描述

x x x 分解质因数结果为 x = p 1 k 1 p 2 k 2 ⋯ p n k n x=p_1^{k_1}p_2^{k_2}\cdots p_n^{k_n} x=p1k1p2k2pnkn,令 f ( x ) = ( k 1 + 1 ) ( k 2 + 1 ) ⋯ ( k n + 1 ) f(x)=(k_1+1)(k_2+1)\cdots (k_n+1) f(x)=(k1+1)(k2+1)(kn+1),求 ∑ i = l r f ( i ) \sum_{i=l}^rf(i) i=lrf(i) 998   244   353 998\,244\,353 998244353 取模的结果。
时间复杂度 O ( n ) O(\sqrt n) O(n )

思路

首先要知道 f ( x ) f(x) f(x) 其实就是 x x x 的约数个数。(每种素因子都有 k i k_i ki 个那么可以取 0 , 1 , . . . , k i 0,1,...,k_i 0,1,...,ki 一共 k i + 1 k_i+1 ki+1 种取法排列组合,由唯一分解定理显然乘积都是不同的)因为题目要求的是区间和,于是正常地转化为前缀和形式 a n s = ∑ i = 1 r f ( x ) − ∑ i = 1 l − 1 f ( x ) ans=\sum_{i=1}^rf(x)-\sum_{i=1}^{l-1}f(x) ans=i=1rf(x)i=1l1f(x) ,设 S ( n ) = ∑ i = 1 n f ( x ) S(n) = \sum_{i=1}^nf(x) S(n)=i=1nf(x),问题转化为了求 S ( n ) S(n) S(n) ,也就是 1 − n 1-n 1n n n n 个数的因数个数之和

1 − n 1-n 1n 中含有因子 i i i 的数一共有 ⌊ n i ⌋ \lfloor\frac ni\rfloor in ,直接用数论分块即可。

代码
int ll, rr, n;
int cal(int l, int r) {
	int ansr = 0, ansl = 0;
	n = rr;
    for(int l=1,r;l<=n;l=r+1){
        r=n/(n/l);
        ansr+=(n/l)*(r-l+1) % P;	
        ansr %= P;
    }
    n = ll;
    for(int l=1,r;l<=n;l=r+1){
        r=n/(n/l);
        ansl+=(n/l)*(r-l+1) % P;	
        ansl %= P;
    }
    ansr = (ansr - ansl + P) % P;
    return ansr;
}
void solve() {
    cin >> ll >> rr;
    ll--;
    cout << cal(ll, rr) << endl;
}

例题2 数论分块

P2303 [SDOI2012] Longge 的问题

题目描述

∑ i = 1 n gcd ⁡ ( i , n ) , n < 1 0 9 \sum\limits_{i=1}^n \gcd(i, n),n<10^9 i=1ngcd(i,n),n<109

思路

注意到 g c d ( i , n ) gcd(i,n) gcd(i,n) 的取值最多只有 n n n 的约数个,所以考虑按取值分类,而且出现 g c d gcd gcd 的题往往都需要想办法凑出 g c d ( i , j ) = 1 gcd(i,j)=1 gcd(i,j)=1,于是我们把原式化为 ∑ d ∣ n d ∑ i = 1 n [ g c d ( i , n ) = d ] = ∑ d ∣ n d ∑ i = 1 n d [ g c d ( i , n d ) = 1 ] = ∑ d ∣ n d ϕ ( n d ) \sum_{d|n}d\sum_{i=1}^n[gcd(i,n)=d ]=\sum_{d|n}d\sum_{i=1}^{\frac nd}[gcd(i,\frac nd)=1]=\sum_{d|n}d\phi(\frac nd) dndi=1n[gcd(i,n)=d]=dndi=1dn[gcd(i,dn)=1]=dndϕ(dn)
可以通过枚举 n n n 的约数直接求,时间复杂度 O ( n ) O(\sqrt n) O(n )

代码
void solve() {
    cin >> n;
    ll ans = 0;
    for(ll i = 1; i * i <= n; i++) {
    	if(n % i == 0) {
	    	ans += i * phi(n/i);
	    	if(n/i!=i) {
	    		ans += (n/i) * phi(i);
	    	}
    	}
    }
    cout << ans << endl;
}

例题3 欧拉函数

GCD-Extreme

题目描述

∑ i = 1 n ∑ j = i + 1 n g c d ( i , j ) , n ≤ 4 × 1 0 6 , T ≤ 100 \sum_{i=1}^n\sum_{j=i+1}^ngcd(i,j),n\leq4\times10^6,T\leq 100 i=1nj=i+1ngcd(i,j),n4×106,T100

思路

首先要知道 ∑ i = 1 n ∑ j = i + 1 n g c d ( i , j ) = ∑ j = 1 n ∑ i = 1 j − 1 g c d ( i , j ) \sum_{i=1}^n\sum_{j=i+1}^ngcd(i,j)=\sum_{j=1}^n\sum_{i=1}^{j-1}gcd(i,j) i=1nj=i+1ngcd(i,j)=j=1ni=1j1gcd(i,j)
然后就比较套路了, ∑ j = 1 n ∑ i = 1 j − 1 g c d ( i , j ) = ∑ d = 1 n ∑ j = 1 n d ∑ i = 1 j − 1 d [ g c d ( i , j ) = 1 ] = ∑ d = 1 n d ∑ j = 1 n d ∑ i = 1 j − 1 [ g c d ( i , j ) = 1 ] \sum_{j=1}^n\sum_{i=1}^{j-1}gcd(i,j)=\sum_{d=1}^n\sum_{j=1}^{\frac nd}\sum_{i=1}^{j-1}d[gcd(i,j)=1]=\sum_{d=1}^nd\sum_{j=1}^{\frac nd}\sum_{i=1}^{j-1}[gcd(i,j)=1] j=1ni=1j1gcd(i,j)=d=1nj=1dni=1j1d[gcd(i,j)=1]=d=1ndj=1dni=1j1[gcd(i,j)=1] = ∑ d = 1 n d ∑ j = 1 n d ϕ ( j ) =\sum_{d=1}^nd\sum_{j=1}^{\frac nd}\phi(j) =d=1ndj=1dnϕ(j)
通过预处理欧拉函数的前缀和以及 1 1 1 n n n 的前缀和,通过数论分块即可以对每组询问 O ( n ) O(\sqrt n) O(n ) 回答,总时间复杂度 O ( T n ) O(T\sqrt n) O(Tn )

代码
//sum[i]表示欧拉函数前i项的和,pre[i]表示1-i的和
void solve() {
    int m = 4e6;
    init(4e6 + 1);
    for(int i = 1; i <= m; i++) {
    	sum[i] = sum[i-1] + phi[i];
    	pre[i] = pre[i-1] + i;
    }
    int n;
    while(cin >> n && n) {
    	ll ans = 0;
    	for(int l = 1, r; l <= n; l = r + 1) {
    		r = n / (n / l);
    		ans += sum[n/l] * (pre[r] - pre[l-1]);
    	}
    	cout << ans << endl;
    }
}

莫比乌斯函数

在这里插入图片描述

常用卷积

ϵ = μ ∗ 1 , I d = ϕ ∗ 1 \epsilon=\mu *1, Id=\phi*1 ϵ=μ1,Id=ϕ1
ϵ ( n ) = ∑ d ∣ n μ ( d ) \epsilon(n) = \sum_{d|n}\mu(d) ϵ(n)=dnμ(d) 。其中 ϵ ( n ) = 1 \epsilon(n)= 1 ϵ(n)=1 当且仅当 n = 1 n=1 n=1,否则 = 0 =0 =0

引例

给定 n , m n, m n,m,求二元组 ( x , y ) (x,y) (x,y) 的个数,满足 1 ≤ x ≤ n 1\leq x\leq n 1xn 1 ≤ y ≤ m 1\leq y\leq m 1ym ,且 g c d ( x , y ) = 1 gcd(x,y)=1 gcd(x,y)=1 n , m ≤ 1 0 7 n,m\leq 10^7 n,m107 ,至多 1 0 4 10^4 104 组数据。

思路

∑ x = 1 n ∑ y = 1 m [ g c d ( x , y ) = 1 ] = ∑ x = 1 n ∑ y = 1 m ϵ ( g c d ( x , y ) ) = ∑ x = 1 n ∑ y = 1 m ∑ d ∣ g c d ( x , y ) μ ( d ) \sum_{x=1}^n\sum_{y=1}^m[gcd(x,y)=1]=\sum_{x=1}^n\sum_{y=1}^m\epsilon (gcd(x,y))=\sum_{x=1}^n\sum_{y=1}^m\sum_{d|gcd(x,y)}\mu(d) x=1ny=1m[gcd(x,y)=1]=x=1ny=1mϵ(gcd(x,y))=x=1ny=1mdgcd(x,y)μ(d)
= ∑ x = 1 n ∑ y = 1 m ∑ d ∣ x , d ∣ y μ ( d ) = = ∑ d = 1 m i n ( n , m ) μ ( d ) ⌊ n d ⌋ ⌊ m d ⌋ =\sum_{x=1}^n\sum_{y=1}^m\sum_{d|x,d|y}\mu(d)==\sum_{d=1}^{min(n,m)}\mu(d)\lfloor\frac nd\rfloor\lfloor\frac md\rfloor =x=1ny=1mdx,dyμ(d)==d=1min(n,m)μ(d)dndm
对上式预处理 μ \mu μ 及其前缀和,按 ⌊ n d ⌋ ⌊ m d ⌋ \lfloor\frac nd\rfloor\lfloor\frac md\rfloor dndm 的取值分段,就可以每组询问 O ( n ) O(\sqrt n) O(n ) 解决问题。

例题4 莫比乌斯反演

题目描述

∑ i = 1 n ∑ j = 1 m i j [ g c d ( i , j ) = k ] , n ≤ m ≤ 4 × 1 0 6 , T ≤ 100 \sum_{i=1}^n\sum_{j=1}^mij[gcd(i, j)=k] ,n\leq m\leq4\times10^6,T\leq 100 i=1nj=1mij[gcd(i,j)=k],nm4×106,T100

思路

在这里插入图片描述

例题5 莫比乌斯反演

题目描述

∑ i = 1 n ∑ j = 1 m l c m ( i , j ) \sum_{i=1}^n\sum_{j=1}^mlcm(i,j) i=1nj=1mlcm(i,j)

思路

n \sqrt n n 的做法,但苯人还没学会。
在这里插入图片描述

例题6 莫比乌斯反演

[SDOI2015]约数个数和

题目描述

d ( x ) d(x) d(x) x x x 的约数个数,给定 n , m n,m n,m,求
∑ i = 1 n ∑ j = 1 m d ( i j ) \sum_{i=1}^n\sum_{j=1}^md(ij) i=1nj=1md(ij)
【数据范围】
对于 100 % 100\% 100% 的数据, 1 ≤ T , n , m ≤ 50000 1\le T,n,m \le 50000 1T,n,m50000

思路

首先有结论:
d ( i j ) = ∑ x ∣ i ∑ y ∣ j [ g c d ( x , y ) = 1 ] d(ij)=\sum_{x|i}\sum_{y|j}[gcd(x,y)=1] d(ij)=xiyj[gcd(x,y)=1]

则原式:
∑ i = 1 n ∑ j = 1 m d ( i j ) = ∑ i = 1 n ∑ j = 1 m ∑ x ∣ i ∑ y ∣ j [ g c d ( x , y ) = 1 ] = ∑ x = 1 n ∑ y = 1 m [ g c d ( x , y ) = 1 ] ⌊ n x ⌋ ⌊ m y ⌋ \sum_{i=1}^n\sum_{j=1}^md(ij)=\sum_{i=1}^n\sum_{j=1}^m\sum_{x|i}\sum_{y|j}[gcd(x,y)=1]=\sum_{x=1}^n\sum_{y=1}^m[gcd(x,y)=1]\lfloor\frac nx\rfloor\lfloor\frac my\rfloor i=1nj=1md(ij)=i=1nj=1mxiyj[gcd(x,y)=1]=x=1ny=1m[gcd(x,y)=1]xnym
将引例中结论带入:
= ∑ d = 1 m i n ( n , m ) μ ( d ) ∑ x = 1 ⌊ n d ⌋ ∑ y = 1 ⌊ m d ⌋ ⌊ n x d ⌋ ⌊ m y d ⌋ = ∑ d = 1 m i n ( n , m ) μ ( d ) ∑ x = 1 ⌊ n d ⌋ ∑ y = 1 ⌊ m d ⌋ ⌊ ⌊ n d ⌋ x ⌋ ⌊ ⌊ m d ⌋ y ⌋ =\sum_{d=1}^{min(n,m)}\mu(d)\sum_{x=1}^{\lfloor\frac nd\rfloor}\sum_{y=1}^{\lfloor\frac md \rfloor}\lfloor\frac n{xd}\rfloor\lfloor\frac m{yd}\rfloor=\sum_{d=1}^{min(n,m)}\mu(d)\sum_{x=1}^{\lfloor\frac nd\rfloor}\sum_{y=1}^{\lfloor\frac md \rfloor}\lfloor\frac{\lfloor\frac nd\rfloor}x\rfloor\lfloor\frac {\lfloor \frac md\rfloor}y\rfloor =d=1min(n,m)μ(d)x=1dny=1dmxdnydm=d=1min(n,m)μ(d)x=1dny=1dmxdnydm
f ( n ) = ∑ i = 1 n ⌊ n i ⌋ f(n)=\sum_{i=1}^n\lfloor\frac ni\rfloor f(n)=i=1nin ,则
= ∑ d = 1 m i n ( n , m ) μ ( d ) f ( ⌊ n d ⌋ ) f ( ⌊ m d ⌋ ) =\sum_{d=1}^{min(n, m)}\mu(d)f(\lfloor\frac nd\rfloor)f(\lfloor\frac md\rfloor) =d=1min(n,m)μ(d)f(dn)f(dm)
f ( n ) = ∑ i = 1 n ⌊ n i ⌋ = ∑ k = 1 n d ( k ) f(n)=\sum_{i=1}^n\lfloor\frac ni\rfloor=\sum_{k=1}^nd(k) f(n)=i=1nin=k=1nd(k)
故可以通过欧拉筛 O ( n ) O(n) O(n) 预处理出 d ( n ) d(n) d(n),通过前缀和得到 f ( n ) f(n) f(n),可以每次询问 O ( n ) O(\sqrt n) O(n ) 回答。

代码
int mu[maxn], prim[maxn], sum[maxn], cnt;
int g[maxn];
bool vis[maxn];
void get_mu(int n)
{
    mu[1]=1;
    for(int i=2;i<=n;i++)
    {
        if(!vis[i]){prim[++cnt]=i;mu[i]=-1;}
        for(int j=1;j<=cnt&&prim[j]*i<=n;j++)
        {
            vis[prim[j]*i]=1;
            if(i%prim[j]==0)break;
            else mu[i*prim[j]]=-mu[i];
        }
    }
    for(int i=1;i<=n;i++)sum[i]=sum[i-1]+mu[i];
    for(int i=1;i<=n;i++)
    {
        long long ans=0;
        for(int l=1,r;l<=i;l=r+1)
        {
            r=(i/(i/l));
            ans+=1ll*(r-l+1)*1ll*(i/l);
        }
        g[i]=ans;
    }
}
void solve() {
    int n, m;
    cin >> n >> m;
    int ans = 0;
    int mn = min(n, m);
    for(int l = 1, r; l <= mn; l = r + 1){
    	r = min(n/(n/l), m/(m/l));
        ans += (sum[r] - sum[l-1]) * g[n/l] * g[m/l];
    }
    cout << ans << endl;
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值