杭电day3 A-深度自同构

题目链接

深度自同构

Problem Description

对于无向图中的点,定义一个点的度为与其相连的边的条数。

对于一棵有根树,定义一个点的深度为该点到根的距离。

对于由若干有根树构成的森林,定义该森林是深度自同构的,当且仅当森林中任意两个深度相同的节点都有相同的度。

请计数有多少个深度自同构的无编号有根树森林,使得其恰好由 n 个节点构成,答案对 998244353 取模。

Input

一个整数 N (1≤N≤10^6),代表计数的范围。

Output

输出一行 N 个整数,第 i 个数代表 n=i 时的答案,对 998244353 取模。

Sample Input

5

Sample Output

1 2 3 5 6

解题思路

刚开始看到这道题的时候能够很快的看出来是一道递推,但确实文字有点晦涩样例有点不好懂(雾)。令 s i 表示 i 个点的合法的树的 个数,枚举根的儿子个数,有 fi = ∑ d|(i−1) fd。这一转移可以用枚举倍数的 方法加速,复杂度为调和级数。 再考虑合法的森林个数,注意到森林中每棵树必定完全相同,因此 ansi = ∑ d|i fd,再用调和级数的复杂度算一遍即可。

#include<bits/stdc++.h>
#define int long long
#define endl '\n'
using namespace std;
const int N = 1e6 + 7;
const int mod = 998244353;
int s[N], ans[N];
signed main() {
	std::ios::sync_with_stdio(0);
	cin.tie(0);
	cout.tie(0);
	int n;
	cin >> n;
	//初始化
	s[1] = 1;
	/*
	令 s i 表示 i 个点的合法的树的个数,
	枚举根的儿子个数,有 fi = ∑ d|(i-1) fd。
	这一转移可以用枚举倍数的 方法加速,
	复杂度为调和级数
	*/
	for (int i = 1; i <= n; i++) {
		for (int j = 1; j * i <= n; j++) {
			(s[j * i + 1] += s[i]) %= mod;
		}
	}
	/*
	我们需要考虑合法的森林个数,
	注意到森林中每棵树必定完全相同,
	因此 ansi = ∑ d|i fd,
	再用调和级数的复杂度算一遍即可
	*/
	for (int i = 1; i <= n; i++) {
		for (int j = i; j <= n; j += i) {
			(ans[j] += s[i]) %= mod;
		}
	}
	int f = 0;
	for (int i = 1; i <= n; i++) {
		if (f)cout << " ";
		f = 1;
		cout << ans[i];
	}
	return 0;
}

 欢迎讨论!!!

  • 7
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值