Coprime

前置芝士

莫比乌斯反演

正文

首先,我们来分析题意。

题目中给出 \(n\) 个人,每个人有一个编号 \(k\) ,要求我们从中选出 \(3\) 个人,三人编号分别为 \(k_a\) ,\(k_b\) ,\(k_c\) ,

使得这三个人的编号满足两两之间同时互质或不互质,最后求出所有满足上述式子的方案。

我们先来考虑:在什么时候,三个数之间的最小公倍数都相等呢?如果我们朴素一点地计算出答案,那么只要用三层循环,并判断条件是否成立,就可以得到答案。但是,这种朴素算法的时间复杂度为 \(O(n^3)\) ,但题目中的 \(n\) 的范围为 \(10^5\) ,所以这样的时间复杂度显然无法帮助我们拿到这道题的满分。

既然朴素算法的时间复杂度,那该如何更快地计算出这道题的答案呢?我们不妨从容斥的角度来考虑。

首先,从 \(n\) 个人中选出 \(k\) 个人,这表示我们最多能有 \(C_n^3\) 种选人方案。然后,我们随机选出三个人,将选出来的三个人之间两两连边,若两个人的编号互质,则连 ** 蓝边  **,若两人的编号不互质,则连  黄边 ,接着我们会发现,若要符合题目条件,我们只能有以下几种可能:

第一种:

第二种:

我们会发现两点:第一:最终所得的三角形的个数即为符合条件的方案数。第二:总有两个人身上有两种颜色的边。所以,我们现在定义一个函数 \(f\) ,\(f(i)\)表示与 \(i\) 互质的数有多少个。那么,在计算三角形个数(即总方案数)时,我们就不用再考虑三个人了,而是只要考虑一个人,不过我们在计算时只能取答案的一半,因为身上连有两条不同边的人有两个。

所以,我们有式子:

\[ANS = C_n^3-\frac {\sum_{i=1}^n f(i) (n-i-f(i))}{2}
\]

式中,\(f(i)\) 表示在范围内与 \(i\) 互质的数,所以 \((n-i-f(i))\) 即为范围内与 \(i\) 不互质的数,而 \(n\) 并非题目中的 \(n\) ,而是题目中的 \(a_i\) 的最大值 (即\(10^5\),接下来我们的 \(n\) 都为 \(10^5\) )。所以,我们只要能求出全部的 \(f(i)\) ,就可以很快地得到答案了。关于 \(f(i)\) ,我们可以通过下面的方法来处理。

在处理 \(f(i)\) 之前,我们先定义一个数组 \(a\) ,\(a_i\) 则表示数字 \(i\) 在数据中是否存在,若存在,则 \(a_i=1\) ,否则 \(a_i=0\)。然后,再根据我们定义的函数 \(f\) ,可以得到下面的式子:

\[f(x)=\sum_{i=1}^n a_i [GCD(i,x)=1]
\]

这算的便是与 i 互质的数的个数了。我们可以继续加以推导:

\[\begin{align}\\
f(x)&=\sum_{i=1}^n a_i [GCD(i,x)=1]\\
&=\sum_{d|x}\mu(d)\sum_{d|i}^n a_i\\
\end{align}\\
\]

然后我们再定义一个函数 \(g\) ,使得 \(g(d)\) =\(\sum_{d|i}^n a_i\) ,这样我们便可以对 \(g\) 进行预处理,从而降低时间复杂度。

预处理的代码如下:

		for (int i=1; i<=n; i++) {
for (int j=i; j<=n; j+=i) g[i]+=a[j];
}

可见这个预处理的时间复杂度为 \(O(n \log n)\) 。

然后,对于 \(\mu\) 的处理,相信大家做到这道题的时候早就不陌生了,这里就直接采用线性筛的方法(其实没有必要)进行预处理了。然后,我们的 \(f\) 便能直接计算出来。然而,如果我们对每个 \(f\) 单独计算,那么时间复杂度就会变成 \(O(n \sqrt {n})\) ,但只要我们同时计算多个 \(f\) ,那么时间复杂度就会降到 \(O(n\log n)\) (关于时间复杂度的计算,请读者自行思考)。而同时处理多个 \(f\) 的也能很容易的实现,如果说原本我们应该枚举的是 \(x\) 的因数,那现在我们可以通过枚举 \(x\) 的倍数来进行优化。

代码如下:

		for (int i=1; i<=n; i++) {
if (mu[i]==0) continue;
for (int j=i; j<=n; j+=i) {
if (a[j]==1) {
f[j]+=mu[i]*(g[i]-1);
}
}
}

最终,我们再将算出来的一切带入最初的式子,就可以计算出答案了。

完整代码如下:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=100005;
int mu[N],prime[N],tot;
bool np[N];
ll f[N],g[N];
bool a[N];
void init(int n) {
mu[1]=1;
np[1]=1;
for (int i=2; i<=n; i++) {
if (!np[i]) {
mu[i]=-1;
prime[tot++]=i;
}
for (int j=0,k; (k=prime[j]*i)<=n; j++) {
np[k]=1;
if (i%prime[j]==0) {
mu[k]=0;
break;
}
mu[k]=-mu[i];
}
}
}
int main() {
int T;
init(100000);
//cout<<1<<endl;
scanf("%d",&T);
for (int _=0; _<T; _++) {
int n,z;
scanf("%d",&n);
z=n;
memset(g,0,sizeof g);
memset(f,0,sizeof f);
memset(a,0,sizeof a);
ll ans=0;
for (int i=1; i<=n; i++) {
int Now;
scanf("%d",&Now);
a[Now]=1;
}
n=100000;
for (int i=1; i<=n; i++) {
for (int j=i; j<=n; j+=i) g[i]+=a[j];
}
for (int i=1; i<=n; i++) {
if (mu[i]==0) continue;//优化
for (int j=i; j<=n; j+=i) {
if (a[j]==1) {
f[j]+=mu[i]*(g[i]-1);
}
}
}
for (int i=1;i<=n;i++) ans=ans+f[i]*(z-1-f[i]);
ans=(ll)z*(z-1)*(z-2)/6-ans/2;
printf("%lld\n",ans);
}
return 0;
}

Coprime的更多相关文章

  1. hdu 5072 Coprime 容斥原理

    Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others)Total Submissio ...

  2. HDU Coprime

    Coprime Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others)Total S ...

  3. HDU 4135 Co-prime(容斥原理)

    Co-prime 第一发容斥,感觉挺有意思的 →_→ [题目链接]Co-prime [题目类型]容斥 &题意: 求(a,b)区间内,与n互质的数的个数. \(a,b\leq 10^{15}\) ...

  4. nyoj CO-PRIME 莫比乌斯反演

    CO-PRIME 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 This problem is so easy! Can you solve it? You are ...

  5. hdu 5072 Coprime (容斥)

    Problem Description There are n people standing in a line. Each of them has a unique id number. Now ...

  6. hdu 4135 Co-prime(容斥)

    Co-prime Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total S ...

  7. Co-prime Array&&Seating On Bus(两道水题)

     Co-prime Array Time Limit:1000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u Su ...

  8. Co-prime(容斥)

    Co-prime Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total S ...

  9. NYOJ 1066 CO-PRIME(数论)

    CO-PRIME 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描写叙述 This problem is so easy! Can you solve it? You are ...

  10. 【转】Hdu--4135 Co-prime

    Problem Description Given a number N, you are asked to count the number of integers between A and B ...

随机推荐

  1. nifi从入门到实战(保姆级教程)——flow

    本文章首发于博客园,转载请标明出处 经过前两篇文章(环境篇,身份验证),我们已经有了nifi可以运行的基础,今天就来实现一个案例吧. 假设我们要从ftp上获取一个zip包,里面有两个csv文件,一个是 ...

  2. 【转载】vscode配置C/C++环境

    VScode中配置 C/C++ 环境 Tip:请在电脑端查看 @零流@火星动力猿 2022.4.12 1. 下载编辑器VScode 官网:https://code.visualstudio.com/( ...

  3. 全国土壤阳离子交换量CEC空间分布数据

    数据下载链接:百度云下载链接​ 土壤阳离子交换量,简称CEC,是指土壤胶体所能吸附各种阳离子的总量.土壤阳离子交换量 cation exchange capacity 即CEC 是指土壤胶体所能吸附各 ...

  4. 一文读懂数仓中的pg_stat

    摘要:GaussDB(DWS)在SQL执行过程中,会记录表增删改查相关的运行时统计信息,并在事务提交或回滚后记录到共享的内存中.这些信息可以通过 "pg_stat_all_tables视图& ...

  5. Entry键值对对象和Map集合遍历键值对方式

    我们已经知道,Map中存放的是两种对象,一种称为key(键),一种称为value(值),它们在在IMap 中是一一对应关系, 这一对对象又称做Map 中的一个Entry(项).Entry将键值对的对应 ...

  6. ConcurrentHashMap树化链表treeifyBin

    private final void treeifyBin(Node<K,V>[] tab, int index) { Node<K,V> b; int n, sc; if ( ...

  7. java中AOP的环绕通知

    pom.xml <dependencies> <dependency> <groupId>org.springframework</groupId> & ...

  8. MySQL主从复制及读写分离

    MySQL主从复制 MySQL数据库自身提供的主从复制功能可以方便的实现数据的多处自动备份,实现数据库的拓展.多个数据备份不仅可以加强数据的安全性,通过实现读写分离还能进一步提升数据库的负载性能. M ...

  9. 工作流引擎在vivo营销自动化中的应用实践 | 引擎篇03

    作者:vivo 互联网服务器团队- Cheng Wangrong 本文是<vivo营销自动化技术解密>的第4篇文章,分析了在营销自动化业务引入工作流技术的背景和工作流引擎的介绍,同时介绍了 ...

  10. PhoneBean实体类的封装和map输出键值对的设置

    之前我们写好了bean类型.现在我们再看看这个需求中,map和reduce各自的流程. Map阶段: 字段切分以后保留如下字段:以第一行为例,就保留13726230503112  2481 24681 ...