zoukankan      html  css  js  c++  java
  • [NOI2016][BZOJ4652][洛谷P1587]循环之美(杜教筛)

    题面

    https://www.luogu.com.cn/problem/P1587

    题解

    前置知识

    题目要求所有(1{leq}x{leq}n,1{leq}y{leq}m)形成的({frac{x}{y}})在k进制下的不同纯循环小数个数。(整数也算)

    分析题目,转化条件:纯循环小数相当于((y,k)=1);而所有值重复的({frac{x}{y}})中,又可以取((x,y)=1)的那个作为代表,这样最好统计。

    所以相当于求

    [{sumlimits_{y=1}^{m}}{sumlimits_{x=1}^{n}}[(y,k)=1][(x,y)=1] ]

    [={sumlimits_{y=1}^{m}}[(y,k)=1]{sumlimits_{x=1}^{n}}{sumlimits_{d|x,d|y}}{mu(d)} ]

    [={sumlimits_{y=1}^{m}}[(y,k)=1]{sumlimits_{d|y}^{1{leq}d{leq}min(n,m)}}{mu(d)}{lfloor}{frac{n}{d}}{ floor} ]

    [={sumlimits_{(d,k)=1}^{1{leq}d{leq}min(n,m)}}{mu(d)}{lfloor}{frac{n}{d}}{ floor}{sumlimits_{y'=1}^{{lfloor}{frac{m}{d}}{ floor}}}[(y',k)=1] ]

    ({mu(d)})后面的东西已经可以通过数论分块处理了。({sum_{y'=1}^{{lfloor}{frac{m}{d}}{ floor}}}[(y',k)=1])这部分,含义是1~({lfloor}{frac{m}{d}}{ floor})中与k互质的数的个数(f({lfloor}{frac{m}{d}}{ floor}))。由于k小,我们可以预处理(f(1))(f(k)),然后将({lfloor}{frac{m}{d}}{ floor})分成k的整数倍和除以k后的余数,就可以(O(1))解决这部分。

    剩下的东西就是如何解决({sum_{(d,k)=1}^{1{leq}d{leq}R}}{mu(d)}),其中R是数论分块后某一块的右端点。

    发现所有的R一定在n或m的整除集合中,所以两次杜教筛就可以解决所有的(summu(R)={sum_{1{leq}d{leq}R}}{mu(d)})。现在的问题是还有一个((d,k)=1)的条件。

    我们考虑({sum_{(d,k){ eq}1}^{1{leq}d{leq}R}}{mu(d)})是什么。然后,惊喜地发现:由于k小,所以既满足((d,k){ eq}1),又满足({mu(d){ eq}0})的数其实不是很多。

    (dt(R))来描述这个和。考虑满足上面一行条件的d和k的gcd,这个gcd只能含有k有的素因子,而且每个次数都不能超过1。但是由于(k{leq}2000),所以k最多只有4个质因子。所以,这样的gcd最多有15个。然后就可以得出关系式:

    [dt(n)={sumlimits_{gcd}}{ }{sumlimits_{(d,k)=1}^{1{leq}d{leq}{frac{k}{gcd}}}}{mu(d)} ]

    [={sumlimits_{gcd}}{mu(gcd)}(summu({frac{k}{gcd}})-dt(frac{k}{gcd})) ]

    所有出现在dt后面的下标都是n或m的整除集合中的数,所以计算所有dt的总复杂度是(O(({sqrt{n}}+sqrt{m})w)),其中w是不同gcd的个数,本题中(w{leq}15)

    综上,本题的总时间复杂度为(O(sqrt{n}w+n^{frac{2}{3}}))

    代码

    #include<bits/stdc++.h>
    
    using namespace std;
    
    #define ll long long
    #define rg register
    #define K 2000
    #define T 1000000
    
    ll n,m,k,pn;
    ll pri[T+5],smu[T+5];
    bool isp[T+5]; 
    
    inline void Eular(){
    	smu[1] = 1;
    	for(rg ll i = 2;i <= T;i++)isp[i] = 1;
    	for(rg ll i = 2;i <= T;i++){
    		if(isp[i])pri[++pn] = i,smu[i] = -1;
    		for(rg ll j = 1;i * pri[j] <= T;j++){
    			isp[i*pri[j]] = 0;
    			if(i % pri[j])smu[i*pri[j]] = -smu[i];
    			else{
    				smu[i*pri[j]] = 0;
    				break;
    			}
    		}
    	}
    	for(rg ll i = 2;i <= T;i++)smu[i] += smu[i-1];
    }
    
    ll f[K+5]; //f[i]表示1~i中与k互质的数的个数 
    
    inline ll gcd(ll a,ll b){
    	return b ? gcd(b,a % b) : a;
    }
    
    inline ll calc(ll n){ //计算1~i中与k互质的数的个数 
    	return (n / k) * f[k] + f[n%k];
    }
    
    unordered_map<ll,ll>Hash;
    
    inline ll summu(int n){ //杜教筛 
    	if(n <= T)return smu[n];
    	if(Hash.count(n))return Hash[n];
    	ll ans = 1;
    	int L,R = 1;
    	while(R < n){
    		L = R + 1,R = n / (n / L);
    		ans -= 1ll * (R - L + 1) * summu(n / L);
    	}
    	return Hash[n] = ans;
    }
    
    vector<ll>fac;
    ll fn,fc[16+5]; //所有k的mu值不等于0的约数(除了1) 
    
    inline void divide(ll n){ //分解k 
    	for(rg ll i = 1;i <= pn;i++){
    		if(n % pri[i] == 0){
    			fac.push_back(pri[i]);
    			while(n % pri[i] == 0)n /= pri[i];
    		}
    		if(n == 1)break;
    	}
    } 
    
    unordered_map<ll,ll>Dt;
    
    inline ll dt(int n){ //1~n中与k互质的数的mu值之和 
    	if(!n)return 0; 
    	if(Dt.count(n))return Dt[n];
    	ll ans = 0;
    	for(rg ll i = 1;i <= fn;i++)ans += (smu[fc[i]] - smu[fc[i]-1]) * (summu(n/fc[i]) - dt(n/fc[i]));
    	return Dt[n] = ans;
    }
    
    int main(){
    	cin >> n >> m >> k;
    	Eular();
    	divide(k);
    	for(rg ll mask = 1;mask < 1ll << fac.size();mask++){
    		ll cur = 1;
    		for(rg ll i = 0;i < fac.size();i++)
    			if((bool)(mask & (1ll << i)))cur *= fac[i];
    		fc[++fn] = cur;
    	}
    	for(rg ll i = 1;i <= k;i++)f[i] = f[i-1] + (gcd(i,k) == 1);
    	ll ans = 0;
    	int L,R = 0,limit = min(n,m);
    	while(R < limit){
    		L = R + 1,R = min(n / (n/L),m / (m/L));
    		ans += ((summu(R)-dt(R)) - (summu(L-1)-dt(L-1))) * (n / L) * calc(m / L);
    	} 
    	cout << ans << endl;
    	return 0;
    }
    
    
  • 相关阅读:
    一、操作m'y's'ql
    十三、并发
    十二、异步_实践
    一、数据
    【2019-08-13】琐碎事才是突破
    【2019-08-12】迟到好过没到
    【2019-08-11】别人约我宵夜,我却约人早茶
    【2019-08-10】习惯跟时间有关
    【2019-08-09】一日之计在于晨
    【2019-08-08】少即是多,慢即是快
  • 原文地址:https://www.cnblogs.com/xh092113/p/12301812.html
Copyright © 2011-2022 走看看