题目链接

定义f[n]表示n是最大公约数情况下的计数,F[n]为n是公约数情况下的计数

(可以和 http://www.cnblogs.com/Just--Do--It/p/7197788.html hdu1695 进行类比)

显然F[n]和f[n]是满足下面这个关系的

所以,可以用下面这个公式求解f[n]

  

得到下面的AC代码

#include<bits/stdc++.h>
using namespace std;
typedef long long LL; #define max(a,b) ((a)>(b)? (a):(b))
#define min(a,b) ((a)<(b)? (a):(b)) const int maxn=1e5+;
int prime[maxn+];
bool check[maxn+];
int mu[maxn+]; void init()
{
mu[]=;
int tot=;
for(int i=;i<=maxn;i++)
{
if(!check[i])
{
prime[tot++]=i;
mu[i]=-;
}
for(int j=;j<tot;j++)
{
if(i*prime[j]>maxn) break;
check[i*prime[j]]=true;
if(i%prime[j]==)
{
mu[i*prime[j]]=;
break;
}
else
{
mu[i*prime[j]]=-mu[i];
}
}
}
} const int N=1e5+;
const int mod=1e9+; int n;
int num[*N];
LL F[N];
LL f[N]; LL qpow(LL x,LL n)
{
LL ret=;
for(;n;n>>=)
{
if(n&) ret=ret*x%mod;
x=x*x%mod;
}
return ret;
} int main()
{
init();
int T;
scanf("%d",&T);
for(int kase=;kase<=T;kase++)
{
memset(num,,sizeof(num));
memset(f,,sizeof(f));
int max_gcd=1e9,max_a=;
LL ans=;
scanf("%d",&n);
for(int i=;i<n;i++)
{
int t;
scanf("%d",&t);
num[t]++;
max_gcd=min(t,max_gcd);
max_a =max(t,max_a );
}
for(int i=;i<*N;i++)
num[i]+=num[i-];
for(int i=;i<=max_gcd;i++)
{
F[i]=;
for(int j=i;j<=max_a;j+=i)
F[i]=F[i]*qpow(j/i,num[j+i-]-num[j-])%mod;
}
// =================================
for(int i=;i<=max_gcd;i++)
for(int j=;i*j<=max_gcd;j++)
f[i]=(f[i]+mu[j]*F[i*j])%mod;
for(int i=;i<=max_gcd;i++)
ans=(ans+f[i])%mod;
// =================================
printf("Case #%d: %lld\n",kase,ans);
}
}

然而!=====所夹的部分可以用一行代码代替!!!!虽然运行时间不会减少多少,不过代码量上优化了很多!

不过这种写法的实质其实可以从容斥原理的角度来考虑,再借用了莫比乌斯函数的性质。

定义:  性质Pi表示i是对象x的一个质因数,  集合Ai表示具有性质Pi的对象的集合

比较容易想到,所有可能的公因数对应的对象计数之和,恰为所有集合的并 中的对象的总个数

那么用容斥原理求 所有集合的并 中的对象的总个数时,奇数个基本集合的交前面的系数是正一,偶数个的是负一。并且,n的因数中包含质因数平方的F[n]在这里面是不需要被计数的。

这样恰好就与莫比乌斯函数的性质产生了联系。

参考博客:  http://blog.csdn.net/acterminate/article/details/76216345

也就变成了

#include<bits/stdc++.h>
using namespace std;
typedef long long LL; #define max(a,b) ((a)>(b)? (a):(b))
#define min(a,b) ((a)<(b)? (a):(b)) const int maxn=1e5+;
int prime[maxn+];
bool check[maxn+];
int mu[maxn+]; void init()
{
mu[]=;
int tot=;
for(int i=;i<=maxn;i++)
{
if(!check[i])
{
prime[tot++]=i;
mu[i]=-;
}
for(int j=;j<tot;j++)
{
if(i*prime[j]>maxn) break;
check[i*prime[j]]=true;
if(i%prime[j]==)
{
mu[i*prime[j]]=;
break;
}
else
{
mu[i*prime[j]]=-mu[i];
}
}
}
} const int N=1e5+;
const int mod=1e9+; int n;
int num[*N];
LL F[N];
LL f[N]; LL qpow(LL x,LL n)
{
LL ret=;
for(;n;n>>=)
{
if(n&) ret=ret*x%mod;
x=x*x%mod;
}
return ret;
} int main()
{
init();
int T;
scanf("%d",&T);
for(int kase=;kase<=T;kase++)
{
memset(num,,sizeof(num));
memset(f,,sizeof(f));
int max_gcd=1e9,max_a=;
LL ans=;
scanf("%d",&n);
for(int i=;i<n;i++)
{
int t;
scanf("%d",&t);
num[t]++;
max_gcd=min(t,max_gcd);
max_a =max(t,max_a );
}
for(int i=;i<*N;i++)
num[i]+=num[i-];
for(int i=;i<=max_gcd;i++)
{
F[i]=;
for(int j=i;j<=max_a;j+=i)
F[i]=F[i]*qpow(j/i,num[j+i-]-num[j-])%mod;
// printf("F[%d]=%4lld\n",i,F[i]);
ans=(ans-mu[i]*F[i]+mod)%mod;
}
printf("Case #%d: %lld\n",kase,ans);
}
}

hdu 6053: TrickGCD (2017 多校第二场 1009) 【莫比乌斯 容斥原理】的更多相关文章

  1. HDU 6053 - TrickGCD | 2017 Multi-University Training Contest 2

    /* HDU 6053 - TrickGCD [ 莫比乌斯函数,筛法分块 ] | 2017 Multi-University Training Contest 2 题意: 给出数列 A[N],问满足: ...

  2. hdu 6055 : Regular polygon (2017 多校第二场 1011) 【计算几何】

    题目链接 有个结论: 平面坐标系上,坐标为整数的情况下,n个点组成正n边形时,只可能组成正方形. 然后根据这个结论来做. 我是先把所有点按照 x为第一关键字,y为第二关键字 排序,然后枚举向量 (p[ ...

  3. hdu 6045: Is Derek lying? (2017 多校第二场 1001)【找规律】

    题目链接 可以暴力找一下规律 比如,假设N=7,两人有5题相同,2题不同,枚举X=0->15时,Y的"Not lying"的取值范围从而找出规律 #include<bi ...

  4. hdu 6047: Maximum Sequence (2017 多校第二场 1003)【贪心】

    题目链接 可以贪心写,先把b数组按从小到大的顺序排个序,根据b[i]的值来产生a[n+i] 借助一个c数组,c[i]记录,j从i到n,a[j]-j的最大值,再加上一个实时更新的变量ma,记录从n+1到 ...

  5. hdu 6050: Funny Function (2017 多校第二场 1006) 【找规律】

    题目链接 暴力打个表找下规律就好了,比赛时看出规律来了倒是,然而看这道题看得太晚了,而且高中的那些数列相关的技巧生疏了好多,然后推公式就比较慢..其实还是自身菜啊.. 公式是 #include< ...

  6. hdu 5308 (2015多校第二场第9题)脑洞模拟题,无语

    题目链接:http://acm.hdu.edu.cn/listproblem.php?vol=44 题意:给你n个n,如果能在n-1次运算之后(加减乘除)结果为24的输出n-1次运算的过程,如果不能输 ...

  7. hdu 5301 Buildings (2015多校第二场第2题) 简单模拟

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5301 题意:给你一个n*m的矩形,可以分成n*m个1*1的小矩形,再给你一个坐标(x,y),表示黑格子 ...

  8. HDU 6053 TrickGCD —— 2017 Multi-University Training 2

    TrickGCD Time Limit: 5000/2500 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others)Total ...

  9. 【HDU 5305】Friends 多校第二场(双向DFS)

    依据题意的话最多32条边,直接暴力的话 2 ^ 32肯定超时了.我们能够分两次搜索时间复杂度降低为 2 * 2  ^ 16 唯一须要注意的就是对眼下状态的哈希处理. 我採用的是 十进制表示法 跑的还是 ...

随机推荐

  1. vb语法

    If IsNull(rs(0)) = False Then picPath = rs(0) End If 判断查询数据库结果集是否为null, 如果用rs(0)则会报错,这样获取不到这条记录的null ...

  2. SynchronousQueue 源码分析

    SynchronousQueue SynchronousQueue 能解决什么问题?什么时候使用 SynchronousQueue? 1)SynchronousQueue 没有任何内部容量. 2)Sy ...

  3. 20160711--C# 委托的三种调用示例(同步调用 异步调用 异步回调)【转载】

    首先,通过代码定义一个委托和下面三个示例将要调用的方法: 代码如下: public delegate int AddHandler(int a,int b); public class 加法类 { p ...

  4. 阶段1 语言基础+高级_1-3-Java语言高级_06-File类与IO流_05 IO字符流_1_字符输入流_Reader类&FileRead

  5. Week13 - 376. Wiggle Subsequence

    Week13 - 376. Wiggle Subsequence A sequence of numbers is called a wiggle sequence if the difference ...

  6. POI读取指定Excel中行与列的数据

    import org.apache.poi.xssf.usermodel.XSSFCell; import org.apache.poi.xssf.usermodel.XSSFRow; import ...

  7. 嗯,python

    总觉得在这么个地方已经没有在碰blog的可能了...但是... 突然说要用python来配置环境...好歹也是这个专业的啊...还是 看看吧... 然后 百度一搜,看到一个 好的 网站,不知道 我一旦 ...

  8. Oracle删除表时候有外键 不能删除

    SELECT    A .constraint_name,    A .table_name,    b.constraint_nameFROM    user_constraints A,    u ...

  9. 编程语言分类,Python代码执行,应用程序使用文件的三步骤,变量,常量,垃圾回收机制

    编程语言分为 机器语言(直接用二进制01跟计算机直接沟通交流,直接操作硬件) 优点:计算机能够直接读懂,速度快 缺点:开发效率极低 汇编语言(用简单的英文标签来表示二进制数,直接操作硬件) 优点:开发 ...

  10. 安全运维 - Windows系统应急响应

    挖矿病毒应急 传播方式: 通过社工.钓鱼方式下载和运行了挖矿程序(邮件.IM等) 利用计算机系统远程代码执行漏洞下载.上传和执行挖矿程序. 利用i算计Web或第三方软件漏洞获取计算机权限,然后下载和执 ...