Description

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

Now the Ragnarok is coming. We should choose 3 people to defend the evil. As a
group, the 3 people should be able to communicate. They are able to communicate
if and only if their id numbers are pairwise coprime or pairwise not coprime.
In other words, if their id numbers are a, b, c, then they can communicate if
and only if [(a, b) = (b, c) = (a, c) = 1] or [(a, b) ≠ 1 and (a, c) ≠ 1 and
(b, c) ≠ 1], where (x, y) denotes the greatest common divisor of x and y.

We want to know how many 3-people-groups can be chosen from the n people.

Input

The first line contains an integer T (T ≤
5), denoting the number of the test cases.

For each test case, the first line contains an integer n(3 ≤ n ≤ 10 5),
denoting the number of people. The next line contains n distinct integers a1,
2, . . . , a n(1 ≤ a i ≤
10 5) separated by a single space, where a i stands
for the id number of the i-th person.

Output

For each test case, output the answer in a
line.

Sample Input

1

5

1 3 9 10 2

Sample Output

4

题目大意是在n个数里面取出三个数a, b, c,如果三者两两互质或者两两都不互质就加入计算。求总的取法数。

首先这两种情况都比较难算,于是考虑补集,其中有且仅有两个互质或者不互质。

这两种情况发现有个共同特征,就是有一个互质且有一个不互质,另外一种关系就随便了。

这样的话就是对于每一个a,计算和它互质的乘上和它不互质的。

但是这样还是有问题的:

因为对于如果(a, b) = 1 && (a, c) != 1,那么如果(b, c) = 1,发现对于数c,同样满足a的条件。

所以对于每一个数都会被计算两次,也就是每一个三元组都被计入了两次,最后结果需要除以2。

接下来就是解决如何计算对于一个a,和a互质以及不互质的数个数。

首先如果可以枚举a的质因子的话,比如p,那么所有p的倍数都是与a不互质的。

但是如果对于q,也加上q的倍数的话,会出现重复,就是pq的倍数被计算了两次。

所以这里计算的时候需要进行容斥。

考虑到这里的话,就不需要枚举a的因子了,只需要枚举所有数找倍数,用容斥判断这里的倍数个数是加还是减, 还是没有贡献。这里的容斥我用莫比乌斯系数完成的。

最后用C(n,
3)-ans/2即可。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <set>
#include <queue>
#include <vector>
#define LL long long using namespace std; const int maxN = ;
int n, d[maxN], top;
LL f[maxN];
int prime[maxN], u[maxN], cnt;
bool vis[maxN]; void mobius()
{
memset(vis, false,sizeof(vis));
u[] = ;
cnt = ;
for(int i = ; i < maxN; i++)
{
if(!vis[i])
{
prime[cnt++] = i;
u[i] = -;
}
for(int j = ; j < cnt && (LL)i*prime[j] < maxN; j++)
{
vis[i*prime[j]] = true;
if(i%prime[j])
u[i*prime[j]] = -u[i];
else
{
u[i*prime[j]] = ;
break;
}
}
}
} void input()
{
memset(d, , sizeof(d));
scanf("%d", &n);
int tmp;
top = ;
for (int i = ; i < n; ++i)
{
scanf("%d", &tmp);
d[tmp]++;
top = max(top, tmp);
}
} void work()
{
LL ans = ;
int num;
memset(f, , sizeof(f));
for (int i = ; i <= top; ++i)
{
num = ;
for (int j = i; j <= top; j += i)
num += d[j];
for (int j = i; j <= top; j += i)
f[j] += u[i]*(-num);
}
for (int i = ; i <= top; ++i)
ans += d[i]*f[i]*(n-f[i]-);
ans = (LL)n*(n-)*(n-)/-ans/;
printf("%I64d\n", ans);
} int main()
{
//freopen("test.in", "r", stdin);
mobius();
int T;
scanf("%d", &T);
for (int times = ; times <= T; ++times)
{
input();
work();
}
return ;
}

ACM学习历程—HDU 5072 Coprime(容斥原理)的更多相关文章

  1. ACM学习历程—HDU 5512 Pagodas(数学)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5512 学习菊苣的博客,只粘链接,不粘题目描述了. 题目大意就是给了初始的集合{a, b},然后取集合里 ...

  2. ACM学习历程—HDU 3915 Game(Nim博弈 && xor高斯消元)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3915 题目大意是给了n个堆,然后去掉一些堆,使得先手变成必败局势. 首先这是个Nim博弈,必败局势是所 ...

  3. ACM学习历程—HDU 5536 Chip Factory(xor && 字典树)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5536 题目大意是给了一个序列,求(si+sj)^sk的最大值. 首先n有1000,暴力理论上是不行的. ...

  4. ACM学习历程—HDU 5534 Partial Tree(动态规划)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5534 题目大意是给了n个结点,让后让构成一个树,假设每个节点的度为r1, r2, ...rn,求f(x ...

  5. ACM学习历程—HDU 3949 XOR(xor高斯消元)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3949 题目大意是给n个数,然后随便取几个数求xor和,求第k小的.(重复不计算) 首先想把所有xor的 ...

  6. hdu 5072 Coprime 容斥原理

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

  7. ACM学习历程—HDU1695 GCD(容斥原理 || 莫比乌斯)

    Description Given 5 integers: a, b, c, d, k, you're to find x in a...b, y in c...d that GCD(x, y) = ...

  8. ACM学习历程—HDU 5317 RGCDQ (数论)

    Problem Description Mr. Hdu is interested in Greatest Common Divisor (GCD). He wants to find more an ...

  9. ACM学习历程—HDU 2112 HDU Today(map && spfa && 优先队列)

    Description 经过锦囊相助,海东集团终于度过了危机,从此,HDU的发展就一直顺风顺水,到了2050年,集团已经相当规模了,据说进入了钱江肉丝经济开发区500强.这时候,XHD夫妇也退居了二线 ...

随机推荐

  1. python 可变参数函数定义* args和**kwargs的用法

    python函数可变参数 (Variable Argument) 的方法:使用*args和**kwargs语法.其中,*args是可变的positional arguments列表,**kwargs是 ...

  2. html5中form表单新增属性以及改良的input标签元素的种类

    在HTML5中,表单新增了一些属性,input标签也有了更多的type类型,有些实现了js才能实现的特效,但目前有些浏览器不能全部支持.下面是一些h5在表单和input标签上的一些改动. <!D ...

  3. GitHub 小试牛刀(踩坑记录)

    首先要在GitHub上创建好远程仓库,把README,LISCENCE,.gitignore三个文件在远程仓库初始化好. 然后在创建本地仓库,先要cd到自己的项目目录下,然后: $ git init ...

  4. 【BZOJ1513】[POI2006]Tet-Tetris 3D 二维线段树

    [BZOJ1513][POI2006]Tet-Tetris 3D Description Task: Tetris 3D "Tetris" 游戏的作者决定做一个新的游戏, 一个三维 ...

  5. Windows下oracle-win-64-11g安装步骤

    一. Oracle 下载 官方下地址 http://www.oracle.com/technetwork/database/enterprise-edition/downloads/index.htm ...

  6. centos 下安装pdo_pgsql 只需一个命令_______yum install php56w-pgsql

    [root@localhost ~]# yum install php56w-pgsql Loaded plugins: fastestmirror, langpacks Repository pgd ...

  7. Python: generator, yield, yield from 详解

    1.Generator Expressions 生成器表达式是用小括号表示的简单生成器标记法: generator_expression ::= "(" expression co ...

  8. bd存储

    var sessionData = new Array();var setSessionData=function(key,val){ if(sessionStorage){ sessionStora ...

  9. 内存写越界导致破环堆结构引起的崩溃问题定位经验[如报错malloc(): memory corruption或free(): invalid next size]

    前段时间开发的一个后端C模块上线后,线上出core,初始时,因为訪问压力不大,所以崩溃是上线3天左右出现的.当时用gdb跟进调用堆栈并检查源代码,发现出core位置的代码沒有啥问题.因为当时开发任务较 ...

  10. LeetCode:移动零【283】

    LeetCode:移动零[283] 题目描述 给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序. 示例: 输入: [0,1,0,3,12] 输出: [1,3 ...