loj#6076「2017 山东一轮集训 Day6」三元组 莫比乌斯反演 + 三元环计数
题目大意:
给定\(a, b, c\),求\(\sum \limits_{i = 1}^a \sum \limits_{j = 1}^b \sum \limits_{k = 1}^c [(i, j) = 1][(j, k) = 1][(i, k) = 1]\)
$a, b, c \leq 5*10^4 $
首先莫比乌斯反演
$Ans = \sum \limits_{i = 1}^a \sum \limits_{j = 1}^b \sum \limits_{k = 1}^c [(i, j) = 1][(j, k) = 1][(i, k) = 1] $
\(= \sum \limits_{i} \sum \limits_{j} \sum \limits_{k} \sum \limits_{x |i \;x|j} \mu(x) \sum \limits_{y|j\;y|k} \mu(y) \sum \limits_{z |i\;z|k} \mu(z)\)
\(= \sum \limits_{x} \sum \limits_{y} \sum \limits_{z} \mu(x) \mu(y) \mu(z) \frac{A}{lcm(x, y)} \frac{B}{lcm(x, z)} \frac{C}{lcm(y, z)}\)
那么考虑计算这个式子
注意到其实有效的三元组\((x, y, z)\)是十分稀少的
我们考虑用一种高效的办法来找到所有的三元组
三元环计数是一个十分便利的算法
如果\(\mu(u), \mu(v) \neq 0, lca(u, v) \leq C\),那么我们连边\((u, v)\)
怎么连边呢?
我们先枚举\(lca(u, v)\),然后枚举\(u\),之后再枚举\(gcd(u, v)\)判断即可
对于有两个数相同的情况和三个数相同的情况进行特判即可
复杂度不会算,反正跑的挺快的
ps:我怎么感觉dfs也能过呢?
#include <bits/stdc++.h>
using namespace std;
#define mp make_pair
#define pii pair <int, int>
#define ri register int
#define rep(io, st, ed) for(ri io = st; io <= ed; io ++)
#define drep(io, ed, st) for(ri io = ed; io >= st; io --)
const int sid = 5e4 + 5;
const int cid = 2e6 + 5;
const int mod = 1e9 + 7;
inline void inc(int &a, int b) { a += b; if(a >= mod) a -= mod; }
inline int mul(int a, int b) { return 1ll * a * b % mod; }
int a, b, c, id, ans, tot;
int mu[sid], pr[sid], nop[sid];
int eu[cid], ev[cid], ew[cid], d[sid], vis[sid], vv[sid];
vector <pii> go[sid];
vector <int> fac[sid];
inline void Init() {
mu[1] = 1;
for (int i = 2; i <= 50000; i ++) {
if (!nop[i]) { pr[++ tot] = i; mu[i] = mod - 1; }
for (int j = 1; j <= tot; j ++) {
int p = i * pr[j];
if(p > 50000) break; nop[p] = 1;
if(i % pr[j] == 0) break; if(mu[i]) mu[p] = mod - mu[i];
}
}
for (ri i = 1; i <= tot; i ++)
for (ri j = pr[i]; j <= 50000; j += pr[i])
fac[j].push_back(pr[i]);
}
inline int gcd(int a, int b) { return b ? gcd(b, a % b) : a; }
inline int lcm(int a, int b) { return 1ll * a * b / gcd(a, b);}
inline void calc() {
if(c < a) swap(a, c);
if(c < b) swap(b, c);
if(b < a) swap(a, b);
for (ri x = 1; x <= a; x ++) // x = y = z
if(mu[x]) inc(ans, mul(mu[x], 1ll * (a / x) * (b / x) * (c / x) % mod));
for (ri L = 1; L <= c; L ++) if(mu[L]) {
int v = fac[L].size();
for (ri S = 0; S <= (1 << v) - 1; S ++) {
int x = 1;
rep(i, 0, v - 1) if(S & (1 << i)) x *= fac[L][i];
if(x > b) continue;
for (ri T = S & (S - 1); ; T = (T - 1) & S) {
int D = 1;
rep(j, 0, v - 1) if(T & (1 << j)) D *= fac[L][j];
int y = 1ll * L * D / x;
if(x > y && y <= a) {
d[x] ++; d[y] ++;
eu[++ id] = x; ev[id] = y; ew[id] = L;
inc(ans, mul(mu[y], 1ll * (a / L) * (b / L) * (c / x) % mod));
inc(ans, mul(mu[y], 1ll * (a / x) * (b / L) * (c / L) % mod));
inc(ans, mul(mu[y], 1ll * (a / L) * (b / x) * (c / L) % mod));
inc(ans, mul(mu[x], 1ll * (a / L) * (b / L) * (c / y) % mod));
inc(ans, mul(mu[x], 1ll * (a / y) * (b / L) * (c / L) % mod));
inc(ans, mul(mu[x], 1ll * (a / L) * (b / y) * (c / L) % mod));
}
if(!T) break;
}
}
}
for (ri i = 1; i <= id; i ++) {
int u = eu[i], v = ev[i];
if(d[u] > d[v]) swap(u, v);
go[u].push_back(mp(v, ew[i]));
}
for (ri x = 1; x <= b; x ++) {
for (auto Y : go[x]) vis[Y.first] = x, vv[Y.first] = Y.second;
for (auto Y : go[x]) for (auto Z : go[Y.first]) if(vis[Z.first] == x) {
static int res = 0, cer = 0;
int y = Y.first, z = Z.first, xy = Y.second, yz = Z.second, xz = vv[z];
res = 0; cer = mul(mu[x], mul(mu[y], mu[z]));
inc(res, 1ll * (a / xy) * ((b / xz) * (c / yz) + (b / yz) * (c / xz)) % mod);
inc(res, 1ll * (b / xy) * ((a / xz) * (c / yz) + (a / yz) * (c / xz)) % mod);
inc(res, 1ll * (c / xy) * ((a / xz) * (b / yz) + (a / yz) * (b / xz)) % mod);
inc(ans, mul(cer, res));
}
}
cout << ans << endl;
}
int main() {
cin >> a >> b >> c;
Init(); calc();
return 0;
}
loj#6076「2017 山东一轮集训 Day6」三元组 莫比乌斯反演 + 三元环计数的更多相关文章
- LOJ #6074. 「2017 山东一轮集训 Day6」子序列
#6074. 「2017 山东一轮集训 Day6」子序列 链接 分析: 首先设f[i][j]为到第i个点,结尾字符是j的方案数,这个j一定是从i往前走,第一个出现的j,因为这个j可以代替掉前面所有j. ...
- loj#6074. 「2017 山东一轮集训 Day6」子序列(矩阵乘法 dp)
题意 题目链接 Sol 设\(f[i][j]\)表示前\(i\)个位置中,以\(j\)为结尾的方案数. 转移的时候判断一下\(j\)是否和当前位置相同 然后发现可以用矩阵优化,可以分别求出前缀积和逆矩 ...
- LOJ#6075. 「2017 山东一轮集训 Day6」重建
题目描述: 给定一个 n个点m 条边的带权无向连通图 ,以及一个大小为k 的关键点集合S .有个人要从点s走到点t,现在可以对所有边加上一个非负整数a,问最大的a,使得加上a后,满足:s到t的最短路长 ...
- Loj #6069. 「2017 山东一轮集训 Day4」塔
Loj #6069. 「2017 山东一轮集训 Day4」塔 题目描述 现在有一条 $ [1, l] $ 的数轴,要在上面造 $ n $ 座塔,每座塔的坐标要两两不同,且为整点. 塔有编号,且每座塔都 ...
- Loj #6073.「2017 山东一轮集训 Day5」距离
Loj #6073.「2017 山东一轮集训 Day5」距离 Description 给定一棵 \(n\) 个点的边带权的树,以及一个排列$ p\(,有\)q $个询问,给定点 \(u, v, k\) ...
- Loj 6068. 「2017 山东一轮集训 Day4」棋盘
Loj 6068. 「2017 山东一轮集训 Day4」棋盘 题目描述 给定一个 $ n \times n $ 的棋盘,棋盘上每个位置要么为空要么为障碍.定义棋盘上两个位置 $ (x, y),(u, ...
- loj #6077. 「2017 山东一轮集训 Day7」逆序对
#6077. 「2017 山东一轮集训 Day7」逆序对 题目描述 给定 n,k n, kn,k,请求出长度为 n nn 的逆序对数恰好为 k kk 的排列的个数.答案对 109+7 10 ^ 9 ...
- LOJ #6119. 「2017 山东二轮集训 Day7」国王
Description 在某个神奇的大陆上,有一个国家,这片大陆的所有城市间的道路网可以看做是一棵树,每个城市要么是工业城市,要么是农业城市,这个国家的人认为一条路径是 exciting 的,当且仅当 ...
- loj#6073. 「2017 山东一轮集训 Day5」距离(树链剖分 主席树)
题意 题目链接 Sol 首先对询问差分一下,我们就只需要统计\(u, v, lca(u, v), fa[lca(u, v)]\)到根的路径的贡献. 再把每个点与\(k\)的lca的距离差分一下,则只需 ...
随机推荐
- python内置模块之unittest测试(五)
系列文章 python模块分析之random(一) python模块分析之hashlib加密(二) python模块分析之typing(三) python模块分析之logging日志(四) pytho ...
- centos6.5 nfs实时共享
一.什么时NFS NFS(Network File System)——网络文件系统,是FreeBSD支持的文件系统中的一种,它允许网络中的计算机之间通过TCP/IP网络共享资源.在NFS的应用中,NF ...
- docker在centos7系统镜像下遇到的一些问题
一.成功安装服务后发现无法启动 报错为:Failed to get D-Bus connection: Operation not permitted 系统为centos7官方版镜像,源和依赖之类的都 ...
- 【转】new对象时,类名后加括号和不加括号的区别
请看测试代码: #include <iostream> using namespace std; // 空类 class empty { }; // 一个默认构造函数,一个自定义构造函数 ...
- python中对列表和循环使用的小练习
#author devilf product_list = [ (), (), (), (), () ] shop_list = [] salary = input('pls enter your s ...
- IntelliJ IDEA 通过GsonFormat插件将JSONObject格式的String 解析成实体
GsonFormat插件主要用于使用Gson库将JSONObject格式的String 解析成实体,该插件可以加快开发进度,使用非常方便,效率高. 插件地址:https://plugins.jetbr ...
- mybatis学习笔记--常见的错误
原文来自:<mybatis学习笔记--常见的错误> 昨天刚学了下mybatis,用的是3.2.2的版本,在使用过程中遇到了些小问题,现总结如下,会不断更新. 1.没有在configurat ...
- 在 Win 7或8 下使用 VirtualBOX 虚拟机安装 OS X 10.11 El Capitan 及 Xcode 7.0
注:本文源自于: http://bbs.feng.com/read-htm-tid-9908410.html _____________________________________________ ...
- 用javascript判断当前是安卓平台还是ios平台
通常判断运行环境都是通过navigator.userAgent if (/android/gi.test(navigator.userAgent)){ // todo : android} if (/ ...
- 将模型.pb文件在tensorboard中展示结构
本文介绍将训练好的model.pb文件在tensorboard中展示其网络结构. 1. 从pb文件中恢复计算图 import tensorflow as tf model = 'model.pb' # ...