考虑到删点操作的实质是指认边的方向。

由于这是一棵树,所以有很好的性质。

我们完全可以以此从树叶开始,往上拓扑进行,按照对某个数的取膜的大小来进行操作。

由此可知,除了 \(1\) 以外,任意 \(2 \leq k\) 都有可能,且只有一种方案。

那么如何判断方案是当下的问题。

考虑到我们的的操作过程,我们发现其实在每个质数的同余系下,有且只有一个答案可能存在。

又由于 \(m = n - 1 = \sum a[i]\),那么我们把 \(m\) 质数分解,对这些质数的同余系进行讨论就好。

同时总方案数为 \(2 ^ {n - 1}\) ,依照容斥原理,那么 \(k = 1\) 时答案为 \(2 ^ {n - 1} - \sum_{i = 2} f[i]\)

#include<iostream>
#include<cstdio>
#include<queue>
#define ll long long
#define N 100005
#define mod 998244353 ll T;
ll n;
ll head[N],cnt;
ll in[N],iin[N];
ll a[N],b[N];
ll f[N]; struct P{
int to,next;
}e[N << 1]; inline void clear(){
for(int i = 1;i <= n;++i)
head[i] = in[i] = iin[i] = a[i] = b[i] = 0,f[i] = 0;
for(int i = 1;i <= cnt;++i)
e[i].to = e[i].next = 0;
cnt = 0; } inline void add(int x,int y){
e[++cnt].to = y;
e[cnt].next = head[x];
head[x] = cnt;
in[y] ++ ;
} inline ll qpow(ll a,ll b){
ll ans = 1;
while(b){
if(b & 1)ans = ans * a % mod;
a = a * a % mod;
b >>= 1;
}
return ans;
} std::queue<int>QWQ; ll vis[N]; inline ll gcd(ll x,ll y){
return (x == 0) ? y : gcd(y % x,x);
} inline ll find(ll x){//找出在该同余系下的答案。
for(int i = 1;i <= n;++i)
iin[i] = in[i],a[i] = b[i] = 0,vis[i] = 0;
for(int i = 1;i <= n;++i)
if(iin[i] == 1)//成为叶子
QWQ.push(i);
while(QWQ.size()){
int u = QWQ.front();
QWQ.pop();
vis[u] = 1;
for(int i = head[u];i;i = e[i].next){
int v = e[i].to;
if(vis[v])continue;
if(a[u] == 0)a[v] = (a[v] + 1) % x,b[v] ++ ;else a[u] = (a[u] + 1) % x,b[u] ++;
iin[v] -- ;
if(iin[v] == 1)
QWQ.push(v);
}
}
ll ans = b[1];
for(int i = 2;i <= n;++i)
ans = gcd(ans,b[i]);
return ans;
} int main(){
scanf("%lld",&T);
while(T -- ){
scanf("%lld",&n);
clear();
for(int i = 1;i <= n - 1;++i){
ll x,y;
scanf("%lld%lld",&x,&y);
add(x,y);
add(y,x);
}
ll m = n - 1;
for(int i = 2;i * i <= m;++i){
if(m % i == 0){
ll si = find(i);
if(si % i == 0)
f[si] = 1 ;
while(m % i == 0 && i != 1)m /= i;
}
}
if(m > 1){
ll si = find(m);
if(si % m == 0)
f[si] = 1 ;
}
f[1] = (f[1] + qpow(2,n - 1)) % mod;
for(int i = 2;i <= n;++i)
f[1] = (f[1] - f[i] + mod) % mod;
for(int i = 1;i <= n;++i)
std::cout<<f[i]<<" ";
puts("");
}
}

CF1554E You的更多相关文章

  1. Involuting Bunny! (2021.8)

      CF1555F & Submission.   Tags:「A.生成树」「B.Tricks」   分类处理询问的 trick:连接两个连通块的边显然合法,先用这些边构建生成森林.发现每条边 ...

随机推荐

  1. 【UE4 C++】抛物线路径、发射轨道相关

    基于UGameplayStatics Blueprint_PredictProjectilePath_ByObjectType 根据 Object Type,算出抛物线的点集合和检测结果 static ...

  2. 热身训练4 Eighty seven

    Eighty seven 简要题意: n个卡片,其中第i个卡片的数值为$a[i]$.一共q次询问,每次询问将删去其中3个卡片(可能删除若干相同的卡片)后,问能否选出10个卡片,数值之和等于87. n≤ ...

  3. GPS与AGPS定位服务

    最近客户反馈车子启动从车库开到地面后,机器定位相对OBD内部定位会慢很多. 机器定位主要依赖定位模块 + AGPS辅助定位. 其中定位模块目前主流支持的有以下三种定位系统. 一.GPS(全球定位系统) ...

  4. Ubuntu 16.04 curl 安装 使用

    curl是利用URL语法在命令行方式显工作的开元文件传输工具. 安装 $ sudo apt install -y curl 使用 $ curl http://www.baidu.com 这是最简单的使 ...

  5. etcd原理详解代码剖析

    1 架构 从etcd的架构图中我们可以看到,etcd主要分为四个部分. HTTP Server: 用于处理用户发送的API请求以及其它etcd节点的同步与心跳信息请求. Store:用于处理etcd支 ...

  6. k8s入坑之路(5)kube-apiserver详解

    API Server kube-apiserver 是 Kubernetes 最重要的核心组件之一,主要提供以下的功能 提供集群管理的 REST API 接口,包括认证授权.数据校验以及集群状态变更等 ...

  7. loadRunner运行场景时,事务数为0或是只显示添加的事务的数

    脚本编辑好后,不要着急到controller去执行,注意查看Run-time Settings(运行是设置)-->General(常规)-->Miscellaneous(其他)中查看Aut ...

  8. offsetX各种值总结

    pageX: 页面X坐标位置 pageY: 页面Y坐标位置 screenX: 屏幕X坐标位置 screenY: 屏幕Y坐标位置 clientX: 鼠标的坐标到页面左侧的距离 clientY: 鼠标的坐 ...

  9. mybatis中批量插入的两种方式(高效插入)

    MyBatis简介 MyBatis是一个支持普通SQL查询,存储过程和高级映射的优秀持久层框架.MyBatis消除了几乎所有的JDBC代码和参数的手工设置以及对结果集的检索封装.MyBatis可以使用 ...

  10. Java学习(十)

    今天学习的是参数的传入,感觉这个和c++差不多. 传一个参数进去,要看这个参数是地址还是一个值,如果是值的话那无论在方法中如何加减,也只是另一个局部变量的事情了,与该参数无关,在原方法中参数的值保持不 ...