zoukankan      html  css  js  c++  java
  • [NOI2010]能量采集

    v.[NOI2010]能量采集

    真正自己做出来的第一道莫反题祭~~~~

    题意:

    \(\sum_{i=1}^n\sum_{j=1}^m(2\gcd(i,j)-1)\)

    开始推式子:

    \(\begin{aligned}\sum_{i=1}^n\sum_{j=1}^m(2\gcd(i,j)-1) & =2\sum_{i=1}^n\sum_{j=1}^m\gcd(i,j)-nm\\ & =2\sum_{d=1}^{\min(n,m)}\sum_{i=1}^n\sum_{j=1}^md[\gcd(i,j)=d]-nm\\ & =2\sum_{d=1}^{\min(n,m)}d\sum_{i=1}^n\sum_{j=1}^m[\gcd(i,j)=d]-nm\\ & =2\sum_{d=1}^{\min(n,m)}d\sum_{i=1}^{n/d}\sum_{j=1}^{m/d}[\gcd(i,j)=1]-nm\\ & =2\sum_{d=1}^{\min(n,m)}d\sum_{i=1}^{n/d}\sum_{j=1}^{m/d}\sum_{x|\gcd(i,j)}\mu(x)-nm\\ & =2\sum_{d=1}^{\min(n,m)}d\sum_{i=1}^{n/d}\sum_{j=1}^{m/d}\sum_{x|i,x|j}\mu(x)-nm\\ & =2\sum_{d=1}^{\min(n,m)}d\sum_{x=1}^{\min(n/d,m/d)}\sum_{i=1}^{n/(dx)}\sum_{j=1}^{m/(dx)}\mu(x)-nm\\ & =2\sum_{d=1}^{\min(n,m)}d\sum_{x=1}^{\min(n/d,m/d)}\mu(x)\sum_{i=1}^{n/(dx)}\sum_{j=1}^{m/(dx)}1-nm\\ & =2\sum_{d=1}^{\min(n,m)}d\sum_{x=1}^{\min(n/d,m/d)}\mu(x)\left\lfloor\dfrac{n}{dx}\right\rfloor\left\lfloor\dfrac{m}{dx}\right\rfloor -nm\end{aligned}\)

    到这里为止,就已经可以做了。

    复杂度为(设\(n\)\(m\)同级)\(O(\sum_{i=1}^n\sqrt{\dfrac{n}{i}})\approx O(n)\)

    但实际上还有\(O(\sqrt{n})\)的做法(虽然预处理是\(O(n)\)的):

    \(2\sum_{d=1}^{\min(n,m)}d\sum_{x=1}^{\min(n/d,m/d)}\mu(x)\left\lfloor\dfrac{n}{dx}\right\rfloor\left\lfloor\dfrac{m}{dx}\right\rfloor -nm\)

    \(T=dx\),则

    \(\begin{aligned}\text{原式} & =2\sum_{T=1}^{\min(n,m)}\left\lfloor\dfrac{n}{T}\right\rfloor\left\lfloor\dfrac{m}{T}\right\rfloor\sum_{d|T}d\mu(\dfrac{T}{d})-mn\end{aligned}\)

    这东西不还是\(O(n)\)的吗?

    我们看一下后面的东西:\(\sum_{d|T}d\mu(\dfrac{T}{d})\)

    我们规定一个函数\(id(x)=x\),则这个东西\(=\sum_{d|T}id(d)\mu(\dfrac{T}{d})\)

    这个运算,有一个具体的名字,叫做狄利克雷卷积。它是这样的运算:

    \(\boxed{h(n)=(f*g)(n)\Leftrightarrow h(n)=\sum_{d|T}f(d)*g(\dfrac{T}{d})}\)

    这个运算,满足:

    \(\boxed{\begin{cases}\text{1.交换律:}f*g=g*f\\\text{2.结合律:}(f*g)*h=f*(g*h)\\\text{3.分配律:}h*(f+g)=h*f+h*g\end{cases}}\)

    它的更多性质我们将在接下来一一提到。不过,在这道题中它最大的作用是帮助我们将上面的东西转成狄利克雷卷积的形式,即

    \(\begin{aligned}\text{原式} & =2\sum_{T=1}^{\min(n,m)}\left\lfloor\dfrac{n}{T}\right\rfloor\left\lfloor\dfrac{m}{T}\right\rfloor((id*\mu)(T))-mn\end{aligned}\)

    好好好,我们搞出了一个新的运算。但是这有什么用吗?

    \(\boxed{\text{如果}f\text{和}g\text{都是积性函数,那么}f*g\text{也是积性函数。}}\)

    是积性函数就意味着我们可以用欧拉筛去筛它!

    而我们已经知道了\(\mu\)函数是积性函数。用脑子稍微想想也知道,\(id\)函数肯定是积性函数。

    这意味着,\(id*\mu\)也是一个积性函数,可以用欧拉筛。

    而这个东西前一半的东西可以整除分块

    那么我们就丧心病狂地把这个东西优化到了\(O(\sqrt{n})\)

    尽管预处理还是\(O(n)\)的,但是已经很优秀了。感觉可以出成一道很毒瘤的题

    先给出普通的\(O(n)\)的代码,而\(O(\sqrt{n})\)的我们将在下一题中用到:

    #include<bits/stdc++.h>
    using namespace std;
    #define int long long
    int n,m,mu[100100],pri[100100],res;
    void getmu(int N){
    	mu[1]=1;
    	for(int i=2;i<=N;i++){
    		if(!pri[i])pri[++pri[0]]=i,mu[i]=-1;
    		for(int j=1;j<=pri[0]&&i*pri[j]<=N;j++){
    			pri[i*pri[j]]=true;
    			if(!(i%pri[j]))break;
    			mu[i*pri[j]]=-mu[i];
    		}
    	}
    	for(int i=1;i<=N;i++)mu[i]+=mu[i-1]; 
    }
    signed main(){
    	scanf("%lld%lld",&n,&m),getmu(min(n,m));
    	for(int i=1;i<=min(n,m);i++){
    		int x=n/i,y=m/i,lim=min(x,y),sum=0;
    		for(int l=1,r;l<=lim;l=r+1)r=min(x/(x/l),y/(y/l)),sum+=(x/l)*(y/l)*(mu[r]-mu[l-1]);
    		res+=2*i*sum;
    	}
    	res-=n*m;
    	printf("%lld\n",res);
    	return 0;
    }
    

  • 相关阅读:
    php中 global 和$GLOBALS有何不同
    perl 使用LDAP模块
    Sqlite和SQLCE在Windows Mobile 6上的性能对比
    Silverlight学习点滴系列(二)
    C#的新特性:自动属性,对象初始化器,和集合初始化器(转载)
    Silverlight学习点滴系列(一)
    Silverlight学习点滴系列(三)
    URL中传递中文参数 以流形式文件上传下载 演变
    C#经典面试题及答案
    C#笔试题目(综合版样题) (转载)
  • 原文地址:https://www.cnblogs.com/Troverld/p/14619536.html
Copyright © 2011-2022 走看看