题意

给定 \(n\) 个点,任意连边,每条边有 \(m\) 种颜色可选,求带环连通图的方案数。

\(1\leq n\leq 10000\)

\(1\leq m < 2^{31}\)

思路

直接求带环连通图显然比较难求,正难则反,考虑容斥。用连通图的个数减去无环连通图(树)的个数。

\(n\) 个节点的无根树,每个节点有区别,可以直接套用公式 \(n^{n-2}\) 。而再考虑边的颜色,就是 \(m^{n-1}n^{n-2}\) 。

我们设 \(n\) 个点,考虑边的颜色,构成不同连通图的方案数为 \(f(n)\) 。

直接求连通图还是不方便,那么我们再容斥:用图的个数减不连通图的个数,\(n\) 个点,考虑边的颜色,可以有 \((m+1)^{n(n+1)\over2}\) 种情况,设之为 \(g(n)\)。

有一个小 \(\text{trick}\) ,我们固定一个点,选一些点和它构成一个连通块,剩下的点任意构图,显然这样是可以不重不漏的,转移式如下
\[ f(n)=g(n)-\sum_{i=1}^{n-1}{n-1\choose i-1}f(i)g(n-i) \]
化简得
\[ f(n)=g(n)-(n-1)!\sum_{i=1}^{n-1}{f(i)\over (i-1)!}\cdot{g(n-i)\over(n-i)!} \]
这样就是一个 \(n^2\) 的 \(dp\) 式,并且形式上满足多项式乘法的形式,只是 \(f\) 在右边出现了。

那我们只能考虑左边对右边的转移,不难想到\(\text{CDQ}\)分治。

void CDQ(int l,int r)
{if(l==r){/*转移常量给dp[l]*/return;}int mid=(l+r)>>1;CDQ(l,mid);/*处理[l,mid]的多项式和转移给[mid+1,r]的多项式*/ _Polynomial::multiply(/**/);/*转移结果给dp[mid+1,r]*/CDQ(mid+1,r);return;
}

代码流程如上,在分治过程中考虑左边转移给右边,需保证在转移前,左边的值以计算完毕。

\(\text{dp}\)式一般写成 \(dp_i=A_i\cdot \sum dp_jf_{i-j}+B_i\) 看的会比较清晰。

代码

#include<bits/stdc++.h>
#define FOR(i,x,y) for(int i=(x),i##END=(y);i<=i##END;++i)
#define DOR(i,x,y) for(int i=(x),i##END=(y);i>=i##END;--i)
using namespace std;
template<typename T,typename _T>inline bool chk_min(T &x,const _T y){return y<x?x=y,1:0;}
template<typename T,typename _T>inline bool chk_max(T &x,const _T y){return x<y?x=y,1:0;}
typedef long long ll;
const int P=152076289;
const int N=1<<14|5;
namespace _Maths
{ll gcd(ll a,ll b){return b?gcd(b,a%b):a;}void exgcd(ll a,ll b,ll &x,ll &y){if(!b){x=1,y=0;return;}exgcd(b,a%b,y,x),y-=a/b*x;}ll Pow(ll a,ll p,ll P){ll res=1;for(;p>0;p>>=1,(a*=a)%=P)if(p&1)(res*=a)%=P;return res;}ll inv(ll a,ll P){ll x,y;exgcd(a,P,x,y);return (x%P+P)%P;}
};
using namespace _Maths;
namespace _Polynomial
{const int g=106;int A[N<<1],B[N<<1];int w[N<<1],r[N<<1];void DFT(int *a,int op,int n){FOR(i,0,n-1)if(i<r[i])swap(a[i],a[r[i]]);for(int i=2;i<=n;i<<=1)for(int j=0;j<n;j+=i)for(int k=0;k<i/2;k++){int u=a[j+k],t=(ll)w[op==1?n/i*k:(n-n/i*k)&(n-1)]*a[j+k+i/2]%P;a[j+k]=(u+t)%P,a[j+k+i/2]=(u-t)%P;}if(op==-1){int I=inv(n,P);FOR(i,0,n-1)a[i]=(ll)a[i]*I%P;}}void multiply(const int *a,const int *b,int *c,int n1,int n2){int n=1;while(n<n1+n2-1)n<<=1;FOR(i,0,n1-1)A[i]=a[i];FOR(i,0,n2-1)B[i]=b[i];FOR(i,n1,n-1)A[i]=0;FOR(i,n2,n-1)B[i]=0;FOR(i,0,n-1)r[i]=(r[i>>1]>>1)|((i&1)*(n>>1));w[0]=1,w[1]=Pow(g,(P-1)/n,P);FOR(i,2,n-1)w[i]=(ll)w[i-1]*w[1]%P;DFT(A,1,n),DFT(B,1,n);FOR(i,0,n-1)A[i]=(ll)A[i]*B[i]%P;DFT(A,-1,n);FOR(i,0,n1+n2-2)c[i]=(A[i]+P)%P;}
};
int A[N],B[N],C[N<<1];
int fac[N],ifac[N],f[N],g[N];
int n;ll m;void CDQ(int l,int r)
{if(l==r){f[l]=(g[l]-(ll)fac[l-1]*f[l]%P)%P;return;}int mid=(l+r)>>1;CDQ(l,mid);FOR(i,l,mid)A[(i)-l]=(ll)f[i]*ifac[i-1]%P;FOR(i,1,r-l)B[(i)-1]=(ll)g[i]*ifac[i]%P;_Polynomial::multiply(A,B,C,mid-l+1,r-l);FOR(i,mid+1,r)f[i]=((ll)f[i]+C[(i)-l-1])%P;CDQ(mid+1,r);
}int main()
{fac[0]=fac[1]=1;FOR(i,2,N-1)fac[i]=(ll)fac[i-1]*i%P;ifac[0]=ifac[1]=1;FOR(i,2,N-1)ifac[i]=(ll)(P-P/i)*ifac[P%i]%P;FOR(i,2,N-1)ifac[i]=(ll)ifac[i-1]*ifac[i]%P;int T;scanf("%d",&T);FOR(Ti,1,T){scanf("%d%lld",&n,&m);FOR(i,1,n)f[i]=0;FOR(i,1,n)g[i]=Pow(m+1,(ll)i*(i-1)/2,P);CDQ(1,n);printf("Case #%d: %lld\n",Ti,(((ll)f[n]-Pow(n,n-2,P)*Pow(m,n-1,P))%P+P)%P);}return 0;
}

转载于:https://www.cnblogs.com/Paulliant/p/10272312.html

HDU 5552 Bus Routes(NTT+分治)相关推荐

  1. HDU 5552 Bus Routes(2015合肥现场赛A,计数,分治NTT)

    题意  给定n个点,任意两点之间可以不连边也可以连边.如果连边的话可以染上m种颜色. 求最后形成的图,是一个带环连通图的方案数. 首先答案是n个点的图减去n个点能形成的树. n个点能形成的树的方案数比 ...

  2. [Leetcode] Bus Routes 公交线路

    Bus Routes 详细解题思路请访问:https://yanjia.me/zh/2018/11/... We have a list of bus routes. Each routes[i] i ...

  3. Leetcode815.公交线路 Bus Routes(Java)

    Leetcode815.公交线路 Bus Routes(Java) ##BFS## 广度优先搜索 2021-03 阿里春招题目 该题应该为图论题,边有权值的情况下寻找最短路径,采用多源BFS解决 示例 ...

  4. (CCPC 2020 网络选拔赛)HDU 6900 Residual Polynomial(分治 + NTT)

    Residual Polynomial 写出所有的fi(x)f_i(x)fi​(x)出来,fi,jf_{i, j}fi,j​表示fi(x)f_i(x)fi​(x)的第jjj项系数 {f1,0f1,1f ...

  5. hdu 5909 Tree Cutting——点分治(树形DP转为序列DP)

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=5909 点分治的话,每次要做一次树形DP:但时间应该是 siz*m2 的.可以用 FWT 变成 siz*ml ...

  6. HDU 4812 D Tree (点分治) (2013ACM/ICPC亚洲区南京站现场赛)

    HDU 4812 D Tree 思路 点对距离相等并且要求输出字典序最小的点对,距离相等不就是点分治裸题了嘛, 照着这个思路出发我们只要记录下所有点对是满足要求的,然后再去找字典序最小的点对就行了, ...

  7. 「LibreOJ NOI Round #2」不等关系 (dp+NTT分治)

    description 戳我看题目哦 solution 有一道非常相似的题目 一棵树,每条边限制两个端点的大小关系(限制 a[u]>a[v]a[u]>a[v]a[u]>a[v] 或 ...

  8. P5644-[PKUWC2018]猎人杀【NTT,分治】

    正题 题目链接:https://www.luogu.com.cn/problem/P5644 题目大意 nnn个人,每个人被选中的权重是aia_iai​.每次按照权重选择一个没有死掉的人杀死,求第11 ...

  9. YbtOJ#903-染色方案【拉格朗日插值,NTT,分治】

    正题 题目链接:https://www.ybtoj.com.cn/contest/115/problem/3 题目大意 两个长度为n+1n+1n+1的序列a,ba,ba,b aia_iai​表示涂了i ...

最新文章

  1. 前端学习(3160):react-hello-react之样式的模块化
  2. 捧上天的AI落地困难,“ 不懂变通”的华为云如何应付?
  3. eclipse的SVN插件设置忽略文件
  4. ASP.NET Gridview的简单的Bootstrap分页
  5. mysql driver pdo_关于 php mysql pdo cannot find driver 解决方案
  6. SWMM与Python的那些事 (一)
  7. 给Photoshop cc 2018添加简体中文语言包
  8. 【数值计算方法】学习笔记
  9. 团队管理13--设定工作目标
  10. Android 官方推荐 : DialogFragment 创建对话框
  11. 比较出名的几种美国主机
  12. 【windows10】将路由器设置为交换机
  13. html做图片模糊效果,CSS3 filter(滤镜) 制作图片高斯模糊无需JS
  14. 如何修改php文件后缀,php如何修改文件后缀名
  15. [源码分析] Facebook如何训练超大模型 --- (3)
  16. new String[0]的作用
  17. 公积金为何会沦为穷人的“鸡肋”?
  18. 什么是RPC?什么是Restful ?它们有什么区别?
  19. 测试beats真假的软件,达人爆料beatsstudio3怎样鉴别真伪?评测不看后悔
  20. 手机电影借山寨机也能闪光

热门文章

  1. python constructor_Python学习札记(2)——搭建Boa-constructor
  2. 闭包 python_Python闭包思想与用法浅析
  3. cdate在java中_Java Calendar.add方法代码示例
  4. jquery修改服务器json,在没有JQuery的情况下将JSON发送到服务器并获取JSON作为回报...
  5. [USACO13FEB]Tractor【二分 + BFS】
  6. 说下自己看源码的一点经验(Python Web 相关)
  7. [Ting's笔记Day9]活用套件Carrierwave gem:(4)使用Imagemagick修改图片大小
  8. Div层悬浮实现HTML5 Canvas背景动画
  9. 微信jssdk配置的问题,使用MVC制作的demo
  10. Android性能优化之UI实战