【题解】CJOI2019 登峰造鸡境 (Prufer序列+斯特林数)

题目背景

  舒服了。

题目描述

你有一颗n个点的无根树,每个点有有一个标号(1~n)。
现在你知道,总共有m个叶子节点,求不同的树的形态方案数。
答案对\(10^9+7\)取模。

下面是一些可能有用的定义:
叶子:度数为1的点。
不同:若对于两颗标号相同的树\(T1=(V,E_1),T2=(V,E_2)\),\(T1\neq T2\)当且仅当存在\((u,v) \in E_1 ,(u,v) \notin E_2\)

输入格式

一共一行,第一行包含两个数n,m分别表示点的总个数和叶子数。
数据保证树一定存在。

输出格式

一行一个整数,输出答案对\(10^9+7\)取模的结果。

输入样例1

5 3

输出样例1

60

子任务

对于\(10\%\)的数据,保证\(n,m<=5\)
对于\(20\%\)的数据,保证\(n,m<=10\)
对于\(40\%\)的数据,保证\(n,m<=20\)
对于\(60\%\)的数据,保证\(n,m<=5000\)
对于另外\(10\%\)的数据,保证\(m=2\)
对于另外\(10\%\)的数据,保证\(m=n-1\)
对于另外\(10\%\)的数据,保证\(m>=n-5\)
对于\(100\%\)的数据,保证\(n,m<=2\times 10^5\)

\(Solution\)

树的计数问题先通过一一对应转换为Prufer序列,再根据Prufer序列和第二类斯特林数求解。

Prufer序列

假设得到一颗有标号的树\(T\),我们通过这样的操作可以得到一个序列,这个序列和它对应的树是一一对应的。也就是说,任何两个不同的合法的Prufer序列都会对应出不同的两颗树。注意到这里的树是带编号的。

​ 在树中,选取一个编号最小的叶子节点,将它的父亲节点加入Prufer序列,并且将这个叶子节点删去。

​ 直到只剩下两个节点为止(只有一条边没有确定了),此时已经可以确定整个树的形态了。

那么得到了一个个数是\(n-2\)个的序列,这个序列和树的形态一一对应。那么这\(n-2\)个元素的序列可以构成
\[
n^{n-2}
\]
种组合。

根据一一对应法则,也就是说有n个不带标号节点的树总共有\(n^{n-2}\)种组合

我们看一下这个序列的意义,一个节点在Prufer序列里出现的次数就是它的度数-1。那么现在问题就变成了,我要保证\(m\)个节点在Prufer序列里不出现。

第二类斯特林数

\(\begin{Bmatrix}n\\m\end{Bmatrix}\)表示\(n\)个元素划分为\(m\)个非空集合的方案数。

这里蕴藏的信息是:元素有区别,集合无区别。

递推公式

\[
{n \brace m}={n-1 \brace m-1}+m{n-1 \brace m}
\]

证明:见yyb博客。

容斥\(O(n)\)或者NTT\(O(n\log n)\)(求一列)

\[
S_2(n,m)=\begin{Bmatrix}n\\m\end{Bmatrix}=\frac 1 {m!} \sum_{i=0}^m (-1)^i{(m-i)^n}{m\choose i}
\]

证明:见yyb博客。

最后的答案
\[
(n-m)!\times{n\choose m}\times{n-2\brace n-m}
\]
因为斯特林数最后的盒子(集合)没有区别,然而我们这里是有区别的,所以应该乘上\((n-m)!\)补回来。

//@winlere
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>

using namespace std;  typedef long long ll;
inline int qr(){
      register int ret=0,f=0;
      register char c=getchar();
      while(c<48||c>57)f|=c==45,c=getchar();
      while(c>=48&&c<=57) ret=ret*10+c-48,c=getchar();
      return f?-ret:ret;
}
const int mod=1e9+7;
const int maxn=2e5+5;
int jc[maxn];
int inv[maxn];
int n,m,ans,k;

typedef const int& ct;
inline int ksm(int base,ct p){
      register int ret=1;
      for(register int t=p;t;t>>=1,base=1ll*base*base%mod)
        if(t&1) ret=1ll*ret*base%mod;
      return ret;
}

int C(int n,int m){
      if(n<m) return 0;
      return 1ll*jc[n]*inv[m]%mod*inv[n-m]%mod;
}

int S(ct n,ct m){
      register int ret=0;
      for(register int t=0;t<=m;++t)
        if(t&1) ret=(0ll+ret-1ll*ksm(m-t,n)*C(m,t)%mod+mod)%mod;
        else ret=(0ll+ret+1ll*ksm(m-t,n)*C(m,t)%mod+mod)%mod;
      return 1ll*ret*inv[m]%mod;
}

int main(){
#ifndef ONLINE_JUDGE
      freopen("dfzjj.in","r",stdin);
      freopen("dfzjj.out","w",stdout);
#endif
      jc[0]=1;
      inv[0]=1;
      for(register int t=1;t<maxn;++t) jc[t]=1ll*jc[t-1]*t%mod,inv[t]=1ll*inv[t-1]*ksm(t,mod-2LL)%mod;
      n=qr();m=qr();
      ans=1ll*jc[n-m]*C(n,m)%mod*S(n-2,n-m)%mod;
      printf("%d\n",ans);
      return 0;
}

【题解】CJOI2019 登峰造鸡境 (Prufer序列+斯特林数)的更多相关文章

  1. bzoj1211: [HNOI2004]树的计数(prufer序列+组合数学)

    1211: [HNOI2004]树的计数 题目:传送门 题解: 今天刚学prufer序列,先打几道简单题 首先我们知道prufer序列和一颗无根树是一一对应的,那么对于任意一个节点,假设这个节点的度数 ...

  2. bzoj1211树的计数 x bzoj1005明明的烦恼 题解(Prufer序列)

    1211: [HNOI2004]树的计数 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3432  Solved: 1295[Submit][Stat ...

  3. 【XSY2519】神经元 prufer序列 DP

    题目描述 有\(n\)点,每个点有度数限制,\(\forall i(1\leq i\leq n)\),让你选出\(i\)个点,再构造一棵生成树,要求每个点的度数不超过度数限制.问你有多少种方案. \( ...

  4. 【BZOJ1211】【HNOI2004】树的计数 prufer序列

    题目描述 给你\(n\)和\(n\)个点的度数,问你有多少个满足度数要求的生成树. 无解输出\(0\).保证答案不超过\({10}^{17}\). \(n\leq 150\) 题解 考虑prufer序 ...

  5. 【XSY1295】calc n个点n条边无向连通图计数 prufer序列

    题目大意 求\(n\)个点\(n\)条边的无向连通图的个数 \(n\leq 5000\) 题解 显然是一个环上有很多外向树. 首先有一个东西:\(n\)个点选\(k\)个点作为树的根的生成森林个数为: ...

  6. prufer序列学习笔记

    prufer序列是一个定义在无根树上的东西. 构造方法是:每次选一个编号最小的叶子结点,把他的父亲的编号加入到序列的最后.然后删掉这个叶节点.直到最后只剩下两个节点,此时得到的序列就是prufer序列 ...

  7. 【BZOJ1005】[HNOI2008]明明的烦恼(prufer序列)

    [BZOJ1005][HNOI2008]明明的烦恼(prufer序列) 题面 BZOJ 洛谷 题解 戳这里 #include<iostream> #include<cstdio> ...

  8. 【CF917D】Stranger Trees 树形DP+Prufer序列

    [CF917D]Stranger Trees 题意:给你一棵n个点的树,对于k=1...n,问你有多少有标号的n个点的树,与给出的树有恰好k条边相同? $n\le 100$ 题解:我们先考虑容斥,求出 ...

  9. BZOJ4766: 文艺计算姬(Prufer序列)

    题面 传送门 题解 结,结论题? 答案就是\(n^{m-1}m^{n-1}\) 我们考虑它的\(Prufer\)序列,最后剩下的两个点肯定是一个在左边一个在右边,设左边\(n\)个点,右边\(m\)个 ...

随机推荐

  1. 代码Rework中的反思

    以前编码只是关注能写出来,并让程序运行就完事,这是非常错误的想法. 让我们重新思考软件设计中的一些问题吧! 软件设计就像设计房屋,设计器具,是一个道理.软件的复杂度和bug完全是自己造成的,要设计好的 ...

  2. pt-query-digest 实践(转)

    mysql slowlog 使用与介绍 slow_query_log =1-----是否打开 slow_query_log_file = /data/mysql_data/node-1/mysql-s ...

  3. 关于Sending build context to Docker daemon 数据很大的问题

    以往进行docker build的时候都是在新建的文件夹下面进行,这次为了图方便,就直接放在开发根目录下进行build,这样子问题就来了.于是就有了下面的文件大小发送量: Sending build ...

  4. 2016.7.12 去除mybatis-generator生成的class里的注释

    用mybatis-generator自动生成代码会出现很多没必要的注释.     在配置文件里加上: 是否去除所有自动生成的文件的时间戳: 是否去除所有自动生成文件的注释: <commentGe ...

  5. angular - 启用form组件

    1.导入form组件 2.导出form组件 3.使用form组件

  6. vue组件class绑定

    当在一个自定义组件上使用 class 属性时,这些类将被添加到该组件的根元素上面.这个元素上已经存在的类不会被覆盖. 例如,如果你声明了这个组件: Vue.component('my-componen ...

  7. UML--组件图,部署图

    组件图用于实现代码之间的物理结构,详细来说,就是实现代码交互.通过接口,将不同的软件,程序连接在一起. [理解] 1.组件的定义相当广泛,包含:源码,子系统,动态链接库,Activex控件. 2.组件 ...

  8. DDR电源硬件设计要点

    一.DDR电源简介 1. 电源 DDR的电源可以分为三类: a.主电源VDD和VDDQ,主电源的要求是VDDQ=VDD,VDDQ是给IO buffer供电的电源,VDD是给但是一般的使用中都是把VDD ...

  9. 一些Python黑客脚本

    [Github项目地址] https://github.com/threeworld/Python

  10. 非标准USBasp下载线烧录Arduino BootLoader的参数设置

    本文仅适用于BootLoader损坏且买到国产“免驱USBasp下载线”导致Arduino IDE无法识别从而不能烧写的情况.是一种略显非主流的操作方式. 因为Arduino的IDE并不支持这种免驱的 ...