zoukankan      html  css  js  c++  java
  • POJ1845 Sumdiv

    Description
    Consider two natural numbers A and B. Let S be the sum of all natural divisors of A^B. Determine S modulo 9901 (the rest of the division of S by 9901).
    Input
    The only line contains the two natural numbers A and B, (0 <= A,B <= 50000000)separated by blanks.
    Output
    The only line of the output will contain S modulo 9901.
    Sample Input
    2 3
    Sample Output
    15
    Hint
    2^3 = 8.
    The natural divisors of 8 are: 1,2,4,8. Their sum is 15.
    15 modulo 9901 is 15 (that should be output).
    题目大意:求ABA^B的约数之和 mod 9901(1<=A,B<=5*10710^7).
    首先,我们需要熟练掌握因式分解。
    1+a+a2++ak1+a+a^2+……+a^k=(ak+11)/(a1)(a^{k+1}-1)/(a-1)
    然后,根据算术基本定理:任何一个数都可以分为若干个(可以为1个)质数的幂之积。
    由此可设:A=p1c1p2c2pkckp_1^{c_1} *p_2^{c_2} *……p_k^{c_k}
    所以A的约数之和为:i=1kpi1+pi2+piciprod_{i=1}^k p_i^1+p_i^2+……p_i^{c_i}=i=1k(pici+11)/(pi1)prod_{i=1}^k (p_i^{c_i+1}-1)/(p_i-1) mod 9901。
    这个有点难理解,设A的约数x=p1d1p2d2pkdkp_1^{d_1} *p_2^{d_2} *……p_k^{d_k},那么一定满足0<=di<=ci
    假设p1p_1的幂不确定,其他质数的幂之积y已经确定,那么约数和就为(p10+p11++p1c1p_1^0 +p_1^1 +……+p_1^{c_1})*y.类似地,我们可以把y继续分解,就可以得到上述公式。(一定要弄懂,不懂可发问)
    我们可设(pici1)=[(pi1)x]9901x+r0&lt;=r&lt;9901(pi1)=[(pi1)9901]x+r(p_i^{c_i}-1)=[(p_i-1)*x]*9901(x为正整数)+r(0&lt;=r&lt;9901*(p_i-1))=[(p_i-1)*9901]*x+r
    那么(pici1)/(pi1)(p_i^{c_i}-1)/(p_i-1) mod 9901=r/(pi1)r/(p_i-1),所以我们只要快速幂求r就行。
    当然,还有一种做法,就是把(pici1)(p_i^{c_i}-1) modmod 99019901 ,和(pi1)(p_i-1)modmod 99019901求出,然后用乘法逆元的方法求出商(用到了费马小定理,因为a/bab99012a/b≡a * b^{9901-2}(mod 9901)(当gcd(b,9901)=1gcd(b,9901)=1))。当gcd(b,9901)!=1时,根据裴蜀定理(即把式子转化成同余方程),不能乘法逆元(得不出整数解),此时b≡1(mod 9901)(即pip_i modmod 99019901=1,pi1+pi2+picip_i^1+p_i^2+……p_i^{c_i}mod 9901=cic_i+1)。
    以上我们求的是A的约数之和,ABA^B的约数之和只需把cic_i均乘以B就行。
    下面贴两个代码:

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    #define ll long long
    const ll mod=9901;
    ll x,y,tot,s[33],b[33];
    // x,y为问题中的a,b
    //tot为质因数种数,s[i]表示第i种质数的个数,b[i]为第i种质数的值
    void divide(ll k)
    {
    	for(ll i=2;i*i<=k;i++)
    	{
    		if(!(k%i))
    		{
    			b[++tot]=i;
    			do
    			{
    				++s[tot];
    				k/=i;
    			}while(!(k%i));
    		}
    	}
    	if(k!=1)b[++tot]=k,++s[tot];
    }
    void mult(ll &a,ll b,ll p)//a*b%p 两种方法 防溢出。注意出题人会千方百计出溢出的数据。 
    {
    	a%=p;b%=p;
    	ll c=(long double)a*b/p;
    	a=a*b-c*p;
    	/*ll ans=0;a%=p;b%=p;
    	while(b)
    	{
    		if(b)ans+=a,ans%=p;
    		b>>=1;a<<=1;a%=p;
    	}
    	return ans;*/
    }
    ll power(ll a,ll b,ll c)//a^b%c
    {
    	ll ans=1%c;a%=c;
    	while(b)
    	{
    		if(b&1)mult(ans,a,c);
    		b>>=1;mult(a,a,c);
    	}
    	return ans;
    }
    
    int main()
    {
    	scanf("%lld%lld",&x,&y);
    	if(x==1){puts("1");return 0;}
    	else if(x==0){puts("0");return 0;}//特判 
    	divide(x);
    	ll ans=1;
    	while(tot)
    	{
    		ll i=b[tot];
    		ans*=((power(i,(y*s[tot--]+1),(i-1)*mod)-1+(i-1)*mod)%((i-1)*mod))/(i-1);
    		ans%=mod;
    	}
    	printf("%lld
    ",ans);
    	return 0;
    }
    
    //不知道为什么这样更快,可能是算模数更快吧。
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    #define ll long long
    const ll mod=9901;
    ll x,y,tot,s[33],b[33];
    // x,y为问题中的a,b
    //tot为质因数种数,s[i]表示第i种质数的个数,b[i]为第i种质数的值
    void divide(ll k)
    {
    	for(ll i=2;i*i<=k;i++)
    	{
    		if(!(k%i))
    		{
    			b[++tot]=i;
    			do
    			{
    				++s[tot];
    				k/=i;
    			}while(!(k%i));
    		}
    	}
    	if(k!=1)b[++tot]=k,++s[tot];
    }
    void mult(ll &a,ll b,ll p)//a*b%p 两种方法 防溢出 
    {
    	a%=p;b%=p;
    	ll c=(long double)a*b/p;
    	a=a*b-c*p;
    	/*ll ans=0;a%=p;b%=p;
    	while(b)
    	{
    		if(b)ans+=a,ans%=p;
    		b>>=1;a<<=1;a%=p;
    	}
    	return ans;*/
    }
    ll power(ll a,ll b,ll c)//a^b%c
    {
    	ll ans=1%c;a%=c;
    	while(b)
    	{
    		if(b&1)mult(ans,a,c);
    		b>>=1;mult(a,a,c);
    	}
    	return ans;
    }
    
    int main()
    {
    	scanf("%lld%lld",&x,&y);
    	if(x==1){puts("1");return 0;}
    	else if(x==0){puts("0");return 0;}//特判 
    	divide(x);
    	ll ans=1;
    	while(tot>0)
    	{
    		ll i=b[tot];//取质数 
    		if((i-1)%mod==0)ans*=s[tot]*y+1;
    		else 
    		{
    			ll aa=(power(i,s[tot]*y+1,mod)-1+mod)%mod,bb=power(i-1,mod-2,mod);//求逆元 
    			ans*=aa*bb;
    		}
    		ans%=mod;--tot;
    	}
    	printf("%lld
    ",ans);
    	return 0;
    }
    
  • 相关阅读:
    RHCE考试要求
    c语言:md5函数
    c语言:计算输入字符个数
    IP数据报之Internet Header Length
    常用的tar和rpm命令参数
    Oracle数据库实例的创建、删除、修改【转载】
    Internal类或Internal成员讲解
    序列化与反序列化 BinaryFormatter.Serialize 方法 (Stream, Object)
    oracle网络配置listener.ora、sqlnet.ora、tnsnames.ora
    Oracle启动模式及其常见问题探讨
  • 原文地址:https://www.cnblogs.com/zsyzlzy/p/12373939.html
Copyright © 2011-2022 走看看