• 【刷题】BZOJ 2693 jzptab


    Description

    Input

    一个正整数T表示数据组数
    接下来T行 每行两个正整数 表示N、M

    Output

    T行 每行一个整数 表示第i组数据的结果

    Sample Input

    1
    4 5

    Sample Output

    122

    HINT

    T <= 10000
    N, M<=10000000

    Solution

    这题就是上一题(BZOJ2154)的升级版
    我们接着推

    [ans=sum_{d=1}^Ndcdot sum_{i=1}^{N}mu(i)cdot i^2cdot s(lfloor frac{N}{id} floor,lfloor frac{M}{id} floor) ]

    (id)提出来

    [ans=sum_{T=1}^Ns(lfloor frac{N}{T} floor,lfloor frac{M}{T} floor)sum_{d|T}frac{T}{d}cdot d^2cdot mu(d) ]

    第一个sigma整除分段,我们主要来考虑第二个sigma
    (H(n)=sum_{d|n}frac{n}{d}cdot d^2cdot mu(d))(F(i)=i)(G(i)=i^2mu(i))
    易知:(F)为积性函数,而(G(ab)=(ab)^2mu(ab)=a^2mu(a)b^2mu(b))(mu)也是积性函数)(=G(a)G(b)),所以(G)也是积性函数
    那么它们卷起来的(H)就也会是积性函数

    1. (H(1)=1)
    2. (H(p,p is a prime)=p-p^2)(直接带进去)
    3. (H(p^a,p is a prime)=sum_{i=0}^ap^{a-i}cdot p^{2i}cdot mu(p^i))
      (i>1)时,根据mu的定义,(mu(p^i))为0,所以(H(p^a,p is a prime)=p^a-p^{a+1})
      这也就是说在素数筛时,对于一个(i*prime[j]),如果(i\%prime[j]!=0),说明(prime[j])是一个新的(i*prime[j])的质数约数,那么(H[i*prime[j]])就加上一个(H[prime[j]])
    4. (H(prod_{i=1} P_i^{a_i} )=prod_{i=1} H(P_i^{a_i}))
      (i\%pime[j]=0),也就是(i*prime[j])中有一质数约数的指数大于1,这个时候因为第三点,它的(mu)是0,不会有额外的贡献,只是(H(i*prime[j]))会乘一个(prime[j])
      (H(P_1^{a_1+1}prod_{i=2}P_i^{a_i})=H(P_1^{a_1+1})H(prod_{i=2}P_i^{a_i}))(积性函数)
      ( =prod_{i=2}(P_i^{a_i}-P_i^{a_i+1})(P_1^{a_1+1}-P_1^{a_1+2}))
      ( =prod_{i=2}(P_i^{a_i}-P_i^{a_i+1})(P_1^{a_1}-P_1^{a_1+1})P_1)
      ( =prod_{i=1}(P_i^{a_i}-P_i^{a_i+1})P_1)
      ( =H(prod_{i=1}P_i^{a_i})P_1)

    后面素数筛搞完后,整除分块,做完
    哦,还有个天大的坑。。。
    mod的数是1e8+9,它也不是质数。。。

    #include<bits/stdc++.h>
    #define ll long long
    const int Mod=1e8+9,MAXN=10000000+10;
    ll s[MAXN],H[MAXN];
    int prime[MAXN],cnt,vis[MAXN];
    template<typename T> inline void read(T &x)
    {
    	T data=0,w=1;
    	char ch=0;
    	while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
    	if(ch=='-')w=-1,ch=getchar();
    	while(ch>='0'&&ch<='9')data=((T)data<<3)+((T)data<<1)+(ch^'0'),ch=getchar();
    	x=data*w;
    }
    template<typename T> inline void write(T x,char c='')
    {
    	if(x<0)putchar('-'),x=-x;
    	if(x>9)write(x/10);
    	putchar(x%10+'0');
    	if(c!='')putchar(c);
    }
    template<typename T> inline void chkmin(T &x,T y){x=(y<x?y:x);}
    template<typename T> inline void chkmax(T &x,T y){x=(y>x?y:x);}
    template<typename T> inline T min(T x,T y){return x<y?x:y;}
    template<typename T> inline T max(T x,T y){return x>y?x:y;}
    inline void init()
    {
    	memset(vis,1,sizeof(vis));
    	vis[0]=vis[1]=0;
    	H[1]=1;
    	for(register ll i=2;i<MAXN;++i)
    	{
    		if(vis[i])
    		{
    			prime[++cnt]=i;
    			H[i]=(i-(ll)i*i)%Mod;
    		}
    		for(register ll j=1;j<=cnt&&i*prime[j]<MAXN;++j)
    		{
    			vis[i*prime[j]]=0;
    			if(i%prime[j])H[i*prime[j]]=H[i]*H[prime[j]]%Mod;
    			else
    			{
    				H[i*prime[j]]=H[i]*(ll)prime[j]%Mod;
    				break;
    			}
    		}
    	}
    	for(register ll	i=1;i<MAXN;++i)s[i]=(s[i-1]+H[i])%Mod;
    }
    inline ll S(ll x,ll y)
    {
    	return ((x+1)*x/2)%Mod*(((y+1)*y/2)%Mod)%Mod;
    }
    inline ll solve(ll N,ll M)
    {
    	ll res=0;
    	if(N>M)std::swap(N,M);
    	for(register ll i=1;;)
    	{
    		if(i>N)break;
    		ll j=min(N/(N/i),M/(M/i));
    		(res+=S(N/i,M/i)*(s[j]-s[i-1])%Mod)%=Mod;
    		i=j+1;
    	}
    	return (res+Mod)%Mod;
    }
    int main()
    {
    	init();
    	int T;
    	read(T);
    	while(T--)
    	{
    		ll N,M;
    		read(N);read(M);
    		write(solve(N,M),'
    ');
    	}
    	return 0;
    }
    
  • 相关阅读:
    java.sql.SQLException: Access denied for user 'root'@'10.1.0.2' (using password: YES)
    在eclipse中安装使用lombok插件
    Socket编程实践(5) --TCP粘包问题与解决
    Socket编程实践(8) --Select-I/O复用
    Socket编程实践(6) --TCP服务端注意事项
    Socket编程实践(4) --多进程并发server
    Socket编程实践(3) --Socket API
    Socket编程实践(2) --Socket编程导引
    Socket编程实践(1) --TCP/IP简述
    Socket编程实践(11) --epoll原理与封装
  • 原文地址:https://www.cnblogs.com/hongyj/p/8562499.html
Copyright © 2020-2023  润新知