考虑令$a_{i}$为i的位置,$p_{i}=0/1$表示第i个点的贡献,那么$p_{x}=0$当且仅当存在与其相邻的点$y$满足$a_{y}<a_{x}$且$p_{y}=1$
树形dp,定义状态$g[k][j][0/1/2]$表示以$k$为根的子树中选择了j个点,$p_{k}=1$或$p_{k}=0$或还没有参与,$f[k][j][0/1/2]$表示这样的$cost$之和
暴力转移即可(转移方程详见代码),注意:1.要乘上组合数表示不同子树之间的顺序;2.要另开一个数组来讨论;3.讨论时要对儿子和根的位置关系分类

  1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 2005
4 #define mod 998244353
5 struct ji{
6 int nex,to;
7 }edge[N];
8 int E,t,n,x,fac[N],inv[N],head[N],sz[N],g[N][N][3],f[N][N][3];
9 int c(int n,int m){
10 return 1LL*fac[n]*inv[m]%mod*inv[n-m]%mod;
11 }
12 void add(int x,int y){
13 edge[E].nex=head[x];
14 edge[E].to=y;
15 head[x]=E++;
16 }
17 void dfs(int k){
18 sz[k]=g[k][0][0]=g[k][0][2]=1;
19 for(int i=head[k];i!=-1;i=edge[i].nex){
20 int u=edge[i].to;
21 dfs(u);
22 for(int j=1;j<sz[u];j++)
23 for(int p=0;p<3;p++){
24 g[u][j][p]=(g[u][j][p]+g[u][j-1][p])%mod;
25 f[u][j][p]=(f[u][j][p]+f[u][j-1][p])%mod;
26 }
27 for(int j=0;j<sz[k];j++)
28 for(int jj=0;jj<=sz[u];jj++){
29 int C=1LL*c(j+jj,j)*c(sz[k]-j-1+sz[u]-jj,sz[u]-jj)%mod;
30 for(int p1=0;p1<3;p1++)
31 for(int p2=0;p2<3;p2++){
32 int s1=0,s2=0;
33 if (jj){
34 s1=1LL*g[u][jj-1][p2]*g[k][j][p1]%mod*C%mod;
35 s2=(1LL*g[u][jj-1][p2]*f[k][j][p1]+1LL*f[u][jj-1][p2]*g[k][j][p1])%mod*C%mod;
36 }
37 if ((p1==0)&&(p2==1)){
38 g[0][j+jj][0]=(g[0][j+jj][0]+s1)%mod;
39 f[0][j+jj][0]=(f[0][j+jj][0]+s2)%mod;
40 }
41 if ((p1==1)&&(p2<2)||(p1==2)&&(p2==0)){
42 g[0][j+jj][1]=(g[0][j+jj][1]+s1)%mod;
43 f[0][j+jj][1]=(f[0][j+jj][1]+s2)%mod;
44 }
45 if ((p1==2)&&(p2==1)){
46 g[0][j+jj][2]=(g[0][j+jj][2]+s1)%mod;
47 f[0][j+jj][2]=(f[0][j+jj][2]+s2)%mod;
48 }
49 if (!jj)s1=g[u][sz[u]-1][p2];
50 else s1=(g[u][sz[u]-1][p2]-g[u][jj-1][p2]+mod)%mod;
51 if (!jj)s2=f[u][sz[u]-1][p2];
52 else s2=(f[u][sz[u]-1][p2]-f[u][jj-1][p2]+mod)%mod;
53 s2=(1LL*s1*f[k][j][p1]+1LL*s2*g[k][j][p1])%mod*C%mod;
54 s1=1LL*s1*g[k][j][p1]%mod*C%mod;
55 if ((p1==0)&&(p2)){
56 g[0][j+jj][0]=(g[0][j+jj][0]+s1)%mod;
57 f[0][j+jj][0]=(f[0][j+jj][0]+s2)%mod;
58 }
59 if ((p1==1)&&(p2<2)){
60 g[0][j+jj][1]=(g[0][j+jj][1]+s1)%mod;
61 f[0][j+jj][1]=(f[0][j+jj][1]+s2)%mod;
62 }
63 if ((p1==2)&&(p2<2)){
64 g[0][j+jj][2]=(g[0][j+jj][2]+s1)%mod;
65 f[0][j+jj][2]=(f[0][j+jj][2]+s2)%mod;
66 }
67 }
68 }
69 sz[k]+=sz[u];
70 for(int j=0;j<sz[k];j++)
71 for(int p=0;p<3;p++){
72 g[k][j][p]=g[0][j][p];
73 f[k][j][p]=f[0][j][p];
74 g[0][j][p]=f[0][j][p]=0;
75 }
76 }
77 for(int i=0;i<sz[k];i++)f[k][i][0]=(f[k][i][0]+g[k][i][0])%mod;
78 }
79 int main(){
80 fac[0]=inv[0]=inv[1]=1;
81 for(int i=1;i<N-4;i++)fac[i]=1LL*fac[i-1]*i%mod;
82 for(int i=2;i<N-4;i++)inv[i]=1LL*(mod-mod/i)*inv[mod%i]%mod;
83 for(int i=2;i<N-4;i++)inv[i]=1LL*inv[i-1]*inv[i]%mod;
84 scanf("%d",&t);
85 while (t--){
86 scanf("%d",&n);
87 E=0;
88 memset(head,-1,4*(n+1));
89 for(int i=1;i<=n;i++){
90 memset(g[i],0,sizeof(g[i]));
91 memset(f[i],0,sizeof(f[i]));
92 }
93 for(int i=2;i<=n;i++){
94 scanf("%d",&x);
95 add(x+1,i);
96 }
97 dfs(1);
98 int ans=0;
99 for(int i=0;i<n;i++)ans=(ans+0LL+f[1][i][0]+f[1][i][1])%mod;
100 printf("%d\n",ans);
101 }
102 }

[nowcoder5668J]Operating on the Tree的更多相关文章

  1. Linux and the Device Tree

    来之\kernel\Documentation\devicetree\usage-model.txt Linux and the Device Tree ----------------------- ...

  2. POJ 2420:A Star not a Tree?

    原文链接:https://www.dreamwings.cn/poj2420/2838.html A Star not a Tree? Time Limit: 1000MS   Memory Limi ...

  3. Device Tree Usage( DTS文件语法)

    http://elinux.org/Device_Tree_Usage Device Tree Usage     Top Device Tree page This page walks throu ...

  4. POJ 2420 A Star not a Tree? 爬山算法

    B - A Star not a Tree? Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/vjudge/co ...

  5. [POJ 2420] A Star not a Tree?

    A Star not a Tree? Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 4058   Accepted: 200 ...

  6. POJ 2420 A Star not a Tree? (计算几何-费马点)

    A Star not a Tree? Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 3435   Accepted: 172 ...

  7. Device Tree Usage(理解DTS文件语法)

    Basic Data Format The device tree is a simple tree structure of nodes and properties. Properties are ...

  8. How to Make a Computer Operating System

    How to Make a Computer Operating System 如何制作一个操作系统(翻译版) 原文地址:Github:How to Make a Computer Operating ...

  9. Spanning Tree Protocol (STP) in NetScaler Appliance

    Spanning Tree Protocol (STP) in NetScaler Appliance 来源 https://support.citrix.com/article/CTX112341 ...

随机推荐

  1. SpringBoot入门07-Thymeleaf中显示ajax请求到的数据

    Thymeleaf中显示ajax请求所需依赖 <!--所需依赖--><dependency> <groupId>org.springframework.boot&l ...

  2. 微信小程序_快速入门02

    01我们学习了环境的准备和简单的demo,现在是时候来学习简单的页面编写了,首先我们来学习一些常用的基础标签: 一.view盒子,就是类似于div的盒子,可以用来存其他元素的容器. 二.text 文本 ...

  3. 小甲鱼零基础学python第25讲课后习题动手练习--通讯录

    小甲鱼零基础学python第25讲课后习题动手练习---通讯录 **************************通讯录要求******************************* 输入指令: ...

  4. 【UE4】读写 Texture 数据

    创建texture 方式一 void AActor_Assignment2::TextureFromImage_Internal( const TArray<FColor>& Sr ...

  5. [Beta]the Agiles Scrum Meeting 8

    会议时间:2020.5.22 21:00 1.每个人的工作 今天已完成的工作 成员 已完成的工作 issue yjy 帮助解决博客评分功能遇到的问题 tq 暂无 wjx 完成批量创建团队项目功能 班级 ...

  6. 基于websocket实现的一个简单的聊天室

    本文是基于websocket写的一个简单的聊天室的例子,可以实现简单的群聊和私聊.是基于websocket的注解方式编写的.(有一个小的缺陷,如果用户名是中文,会乱码,不知如何处理,如有人知道,请告知 ...

  7. VirtualBox Share Folder

    转载:https://www.cnblogs.com/Dennis-mi/articles/5896586.html 使用virtualbox最方便的host-guest交换文件方案莫过于共享文件夹功 ...

  8. diff 命令,防止遗忘

    常规输出: diff 1.file 2.file 并排格式输出: diff 1.file 2.file -y -W 50 显示说明 "|"表示前后2个文件内容有不同 "& ...

  9. Notepad++ 过滤注释行和空行

    Notepad++ 删除指定字符开头的行的正则表达式 1.删除A之后的所有字符用:A.*$ 2.删除A之前的所有字符用:^([^s]*)A ####如果是其他字符就把A替换为其他字符 注释:如何是特殊 ...

  10. 【微服务落地】服务间通信方式: gRPC的入门

    gRPC是什么 官方介绍: https://grpc.io/docs/what-is-grpc/introduction/ "A high-performance, open-source ...