洛谷 P3244 / loj 2115 [HNOI2015] 落忆枫音 题解【拓扑排序】【组合】【逆元】
组合计数的一道好题。什么非主流题目
题目背景
(背景冗长请到题目页面查看)
题目描述
不妨假设枫叶上有 n 个穴位,穴位的编号为 1∼n。有若干条有向的脉络连接着这些穴位。穴位和脉络组成一个有向无环图——称之为脉络图(例如图 1),穴位的编号使得穴位 1 没有从其他穴位连向它的脉络,即穴位 1 只有连出去的脉络;由上面的故事可知,这个有向无环图存在一个树形子图,它是以穴位 1 为根的包含全部 n 个穴位的一棵树——称之为脉络树(例如图 2 和图 3 给出的树都是图 1 给出的脉络图的子图);值得注意的是,脉络图中的脉络树方案可能有多种可能性,例如图 2 和图 3 就是图 1 给出的脉络图的两个脉络树方案。
脉络树的形式化定义为:以穴位 r 为根的脉络树由枫叶上全部 n 个穴位以及 n−1 条脉络组成,脉络树里没有环,亦不存在从一个穴位连向自身的脉络,且对于枫叶上的每个穴位 s,都存在一条唯一的包含于脉络树内的脉络路径,使得从穴位 r 出发沿着这条路径可以到达穴位 s。
现在向脉络图添加一条与已有脉络不同的脉络(注意:连接 2 个穴位但方向不同的脉络是不同的脉络,例如从穴位 3 到 4 的脉络与从 4 到 3 的脉络是不同的脉络,因此,图 1 中不能添加从 3 到 4 的脉络,但可添加从 4 到 3 的脉络),这条新脉络可以是从一个穴位连向自身的(例如,图 1 中可添加从 4 到 4 的脉络)。原脉络图添加这条新脉络后得到的新脉络图可能会出现脉络构成的环。
请你求出添加了这一条脉络之后的新脉络图的以穴位 1 为根的脉络树方案数。
由于方案可能有太多太多,请输出方案数对 1000000007 取模得到的结果。
输入格式
输入文件的第一行包含四个整数 n、m、x 和 y,依次代表枫叶上的穴位数、脉络数,以及要添加的脉络是从穴位 x 连向穴位 y 的。
接下来 m 行,每行两个整数,由空格隔开,代表一条脉络。第 i 行的两个整数为 ui 和 vi,代表第 i 条脉络是从穴位 ui 连向穴位 vi 的。
输出格式
输出一行,为添加了从穴位 x 连向穴位 y 的脉络后,枫叶上以穴位 1 为根的脉络树的方案数对 1000000007 取模得到的结果。
输入输出样例
输入样例:
4 4 4 3 1 2 1 3 2 4 3 2
输出样例:
3
数据范围与约定
对于所有测试数据,1≤n≤100000, n−1≤m≤min(200000,n(n−1)2), 1≤x,y,ui,vi≤n。
题解:
首先需要找出一个不需要拓扑排序就能解决不加边时的脉络树数量的方法。
对于每个点 u(u≥2) ,假定它的入度为 di ,则它有 di 个父亲可供选择。我们只需要从上往下看,就可以发现每一层都是互相独立的,因此加边之前脉络树的数量为
此时考虑加边。正常情况下按上面的方式计数,边数为 n−1 的图的总数 sum 为
但是实际上不是所有 sum 种方案都符合题意,由于每个点选择父亲是自由从入边选的,因此可能存在环,此时就不满足”脉络树“的定义了,而且图/树也没有明显分层。
我们考虑所有的 sum 种方案,从中减掉包含环的那些方案。由于我们加入的边是 ⟨x,y⟩,所以一定是与路径 y→x 成环。因此我们只需要排除那些包含 y→x 的路径的边数为 n−1 的图就可以了。
注意由于加了新边之后的图只有一个环,因此 n−1 条边的图也最多只有一个环。
话再说回来,包含 y→x 的路径的图我们可以认为这条路径上的所有点(包括 x,y)都被钦定了一个父亲(其中 x 的父亲认为是 y,因为要成环)。假设用 S={ai} 表示这条路径,那么包含 y→x 的图种类数就是 ∏i∉Sdi。
而最终的答案就是 sum−∑S:y→x∏i∉Sdi。
此时考虑如何求出所有的 y→x。可以建立原图的反向边,跑拓扑排序。设定状态 fk 表示 ∑S:k→x∏i∉Sdi,每次从原图的边 ⟨u,v⟩ 转移时,即钦定了 v 的入边,所以状态转移方程为
由于我们还要钦定 x 的入边是 y,因此最终的答案是
时间复杂度为 O(n) 或 O(nlogn) (在线求逆元)
Code:
#include<cstdio>
#include<cstring>
#define p 1000000007
int Plus(int x,int y)
{return (x+y>=p)?(x+y-p):(x+y);}
int Mul(int x,int y)
{return 1ll*x*y%p;}
struct edge
{
int n,nxt;
edge(int n,int nxt)
{
this->n=n;
this->nxt=nxt;
}
edge(){}
}e[200000];
int head[100100],ecnt=-1;
void add(int from,int to)
{
e[++ecnt]=edge(to,head[from]);
head[from]=ecnt;
}
int d[100100],in[100100];
//d表示真实入度 in表示拓扑排序中的入度
int q[100100],l=0,r=0;
int f[100100],inv[100100];
int main()
{
memset(head,-1,sizeof(head));
inv[1]=1;
for(int i=2;i<=100000;++i)
inv[i]=Mul(p-p/i,inv[p%i]);
int n,m,x,y,u,v;
scanf("%d%d%d%d",&n,&m,&x,&y);
for(int i=1;i<=m;++i)
{
scanf("%d%d",&u,&v);
add(v,u);
++in[u];
++d[v];
}
f[x]=1;
int sum=1;
for(int i=2;i<=n;++i)
{
if(!in[i])
q[++r]=i;
f[x]=Mul(f[x],d[i]);
sum=Mul(sum,d[i]+(i==y));
}
if(y==1)
{
printf("%d\n",sum);
return 0;
}
while(l<r)
{
int k=q[++l];
for(int i=head[k];~i;i=e[i].nxt)
{
--in[e[i].n];
f[e[i].n]=Plus(f[e[i].n],Mul(f[k],inv[d[k]]));
if(!in[e[i].n])
q[++r]=e[i].n;
}
}
printf("%d\n",Plus(sum,p-Mul(f[y],inv[d[y]])));
return 0;
}
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Linux系统下SQL Server数据库镜像配置全流程详解
· 现代计算机视觉入门之:什么是视频
· 你所不知道的 C/C++ 宏知识
· 聊一聊 操作系统蓝屏 c0000102 的故障分析
· SQL Server 内存占用高分析
· 盘点!HelloGitHub 年度热门开源项目
· DeepSeek V3 两周使用总结
· 02现代计算机视觉入门之:什么是视频
· C#使用yield关键字提升迭代性能与效率
· 回顾我的软件开发经历(1)