当前位置: 首页 > news >正文

题解:AT_abc428_g [ABC428G] Necklace

分享一种比较暴力的做法。

首先肯定是使用 Burnside 引理求解,不过题目并没有给定环的大小,但是由于大小为 \(n\) 的环至少要有 \(2^n\) 的美丽值,所以这个 \(n\) 只有 \(\log m\) 个。

所以可以快乐的枚举环的大小,假设是 \(n\)

那么要求:

\[\sum_{i = 0}^{n - 1}{f(g_i)} \]

\(g_i\) 表示将环顺时针转 \(i\) 下,\(f(g_i)\) 表示在 \(g_i\) 作用下不变的环的个数。

将环上一个点向转完后的位置连边,会连出 \(\gcd(n, i)\) 个环,并且这些环内的颜色必须一样,再满足美丽值之积为 \(x\)

先将 \(a_k\) 变为 \(a_k^{\frac{n}{\gcd(n, i)}}\)

此时设 \(h(i)\) 表示 \(i\) 出现的次数,\(dp_{i, j}\) 表示 \(i\) 个数乘积为 \(j\) 的方案数。

转移为:

\[dp_{i, j} = \sum_{d | j}{h(\frac{j}{d})\times dp_{i - 1, d}} \]

如果设 \(F * G\) 表示两者的狄利克雷卷积,就有:

\[dp_i = dp_{i - 1} * h = h^i \]

\(x\) 的答案就是 \(dp_{\gcd(n, i)}\) 的第 \(x\) 项。

这样长度为 \(n\) 的答案就是 \(\sum_{i = 0}^{n - 1}{h^{\gcd(n, i)}}\)

不过复杂度有点劣,改为枚举 \(gcd(n, i)\),答案变成 \(\sum_{d | n}{h^d \varphi(\frac{n}{d})}\)

总答案为:

\[\sum_{n = 1}^{\log m}{\sum_{d | n}{h^d \varphi(\frac{n}{d})}} \]

注:此处 \(h\) 随着 \(n\)\(d\) 变化。

分析一下复杂度,这两个求和是 \(\mathcal{O}(\sqrt{\log m}\log m)\) 的,卷积使用快速幂会进行 \(\log{\log m}\) 次卷积,一次卷积是 \(\mathcal{O}(m \log m)\) 的。

所以复杂度是 \(\mathcal{O}(m\sqrt{\log m}\log^2 m\log{\log m})\) 的,常数比较小。

#include <iostream>
#include <cstring>
#include <algorithm>using namespace std;const int N = 5e5 + 10, mod = 998244353;#define int long long#define fi first
#define se secondusing pii = pair <int, int>;int n, m, a[N];class Poly
{public :int f[N];friend Poly operator * (Poly x, Poly y){Poly res;for (int i = 1; i <= m; i++) res.f[i] = 0;for (int i = 1; i <= m; i++){if (!x.f[i]) continue;for (int j = 1; j <= m / i; j++){(res.f[i * j] += x.f[i] * y.f[j]) %= mod;}}return res;}
};Poly qpow(Poly x, int b)
{Poly res;for (int i = 1; i <= m; i++) res.f[i] = 0;res.f[1] = 1;while (b){if (b & 1) res = res * x;x = x * x;b >>= 1;}return res;
}int qpow(int x, int b, int lim)
{int res = 1;while (b){if (b & 1){if (x > lim) return false;res = res * x;if (res > lim) return false;}if (x <= lim) x = x * x;b >>= 1;}return res;
}int qpow(int x, int b)
{int res = 1;while (b){if (b & 1) res = res * x % mod;x = x * x % mod;b >>= 1;}return res;
}int phi(int x)
{int res = x;for (int i = 2; i * i <= x; i++){if (x % i) continue;res = res / i * (i - 1);while (x % i == 0) x /= i;}if (x != 1) res = res / x * (x - 1);return res;
}int ans[N];
Poly F;signed main()
{ios :: sync_with_stdio(false), cin.tie(0), cout.tie(0);cin >> n >> m;for (int i = 1; i <= n; i++) cin >> a[i];sort(a + 1, a + 1 + n);int lim = 0;while ((1 << lim) <= m) lim++;for (int _n = 1; _n <= lim; _n++){bool flag = 0;for (int i = n; i >= 1; i--) if (qpow(a[i], _n, m)) flag = 1;if (!flag) break;int inv = qpow(_n, mod - 2);for (int i = 1; i <= _n; i++){if (_n % i) continue;for (int j = 1; j <= m; j++) F.f[j] = 0;for (int j = 1; j <= n; j++) F.f[qpow(a[j], _n / i, m)]++;Poly tmp = qpow(F, i);int ph = phi(_n / i);for (int j = 2; j <= m; j++) (ans[j] += tmp.f[j] * ph % mod * inv) %= mod;}}for (int i = 2; i <= m; i++) cout << ans[i] << ' ';return 0;
}
http://www.zskr.cn/news/53487.html

相关文章:

  • 第十四天 mysql单表练习
  • 题解:P14435 [JOISC 2013] 收拾吉祥物 / Mascots
  • lvs详细配置
  • linux c 线程池
  • linux c 文件是否存在
  • 11月18日
  • 三维偏序整体二分?
  • MEMS与CMOS的3D集成技术研究进展 - 指南
  • 2025 年 钢丝网/钢骨架 塑料复合管厂家权威推荐榜/哪家好/有实力/可靠的/排名企业-江苏狼博管道制造有限公司
  • CSS实现修改CheckBox样式
  • 查看laya已经加载的资源
  • ESP32 + LVGL 开发笔记(一):点亮屏幕
  • linux c makefile
  • 基于自适应遗传算法风光场景生成的电动汽车并网优化调度【IEEE33节点】(Matlab代码建立)
  • High Frequency Active Auroral Research Program(HAARP)部分摘取
  • CF813E Army Creation
  • 铭记旧友
  • update 锁表了: 执行一个update 表被锁了,原因是什么?
  • 标题:鸿蒙Next音频开发新篇章:深入解析Audio Kit(音频服务) - 实践
  • 人工智能之编程进阶 Python高级:第一章 栈和队列
  • DS trick record 2
  • 详细介绍:MonkeyCode:开源AI编程助手的技术实践与应用价值
  • 福利MegaLLM–175刀免费额度建教程
  • 白嫖MegaLLM–175刀免费额度建教程
  • 如何找到适合好用的 AI 数据分析工具?Aloudata Agent 值得一试!
  • linux bug
  • linux broadcom
  • Duan.ai - 将长视频变成适合社交的短视频AI工具
  • 2025年11月成都房产律师,成都合同纠纷律师,成都刑事律师事务所推荐,实力律所解析委托无忧之选!
  • Balatro GBA - 在Game Boy Advance上体验扑克 Roguelike