生成函數
概念
又稱母函數把一個無窮數列 {an}\{a_n\}{an?}(默認從 000 項起)表示成 G(x)=∑i≥0aixiG(x)=\displaystyle\sum_{i\ge0} a_ix^iG(x)=i≥0∑?ai?xi 的函數形式。例如:
- ai=2ia_i=2^iai?=2i:G(x)=∑i≥02ixiG(x)=\displaystyle\sum_{i\ge 0} 2^i x^iG(x)=i≥0∑?2ixi;
- ai=1a_i=1ai?=1:G(x)=∑i≥0xiG(x)=\displaystyle\sum _{i\ge 0} x^iG(x)=i≥0∑?xi。
應用
用來幫助推導數列的通項公式(比如斐波那契數列)也可以作為條件形式以數列形式給出的象征。一個特別的轉化:逆運用等比數列求和公式,可以得到:11?ax=∑i≥0aixi\displaystyle\dfrac{1}{1-ax}=\sum_{i\ge0}a^ix^i1?ax1?=i≥0∑?aixi,等號左邊不帶求和符號部分的,稱為生成函數的封閉形式。
生成函數的乘法運算有意義,其卷積形式是函數直接相乘。對于 A(x)=∑i≥0aixiA(x)=\displaystyle\sum_{i\ge0}a_ix^iA(x)=i≥0∑?ai?xi 和 A(x)=∑i≥0bixiA(x)=\displaystyle\sum_{i\ge0}b_ix^iA(x)=i≥0∑?bi?xi:
A×B=∑i≥0n(xi∑j=0iajbi?j)A\times B=\sum_{i\ge0}^n\left(x^i\sum_{j=0}^ia_jb_{i-j}\right)A×B=i≥0∑n?(xij=0∑i?aj?bi?j?)
洛谷 P10780 BZOJ3028 食物
題意
題目傳送門。
思路
根據上文,將這個背包問題轉化為,考慮每一種物品拿的個數的生成函數,并用封閉形式表示:
- 偶數個形如 {1,0,1,0,1,......}\{1,0,1,0,1,......\}{1,0,1,0,1,......}:∑k≥0x2k=11?x2\displaystyle\sum_{k\ge 0} x^{2k}=\frac{1}{1-x^2}k≥0∑?x2k=1?x21?;
- 000 或 111 個形如 {1,1,0,0,0,......}\{1,1,0,0,0,......\}{1,1,0,0,0,......}:1+x1+x1+x;
- 000 或 111 或 222 個形如 {1,1,1,0,0,......}\{1,1,1,0,0,......\}{1,1,1,0,0,......}:1+x+x21+x+x^21+x+x2;
- 奇數個形如 {0,1,0,1,0,......}\{0,1,0,1,0,......\}{0,1,0,1,0,......}:∑k≥0x2k+1=x∑k≥0x2k=x1?x2\displaystyle\sum_{k\ge 0}x^{2k+1}=x\displaystyle\sum_{k\ge 0} x^{2k}=\frac{x}{1-x^2}k≥0∑?x2k+1=xk≥0∑?x2k=1?x2x?;
- 444 的倍數個類似第一條:∑k≥0x4k=11?x4\displaystyle\sum_{k\ge 0}x^{4k}=\frac{1}{1-x^4}k≥0∑?x4k=1?x41?;
- 000 或 111 或 222 或 333 個形如 {1,1,1,0,0,......}\{1,1,1,0,0,......\}{1,1,1,0,0,......}:1+x+x2+x31+x+x^2+x^31+x+x2+x3;
- 333 的倍數個類似第一條:∑k≥0x3k=11?x3\displaystyle\sum_{k\ge 0}x^{3k}=\frac{1}{1-x^3}k≥0∑?x3k=1?x31?.
全部乘起來,并用二項式定理化簡 1+x+x21+x+x^21+x+x2 和 1+x+x2+x31+x+x^2+x^31+x+x2+x3:
x(1?x)4\frac{x}{(1-x)^4}(1?x)4x?
考慮推 1(1?x)k=(11?x)k\dfrac{1}{(1-x)^k}=\left(\dfrac{1}{1-x}\right)^k(1?x)k1?=(1?x1?)k 是哪個生成函數的封閉形式與該生成函數的第 nnn 項系數,因為 11?ax=∑i≥0aixi\displaystyle\dfrac{1}{1-ax}=\sum_{i\ge0}a^ix^i1?ax1?=i≥0∑?aixi,a=1a=1a=1 時,11?x=1+x+x2+x3+...\dfrac{1}{1-x}=1+x+x^2+x^3+...1?x1?=1+x+x2+x3+...。kkk 次方展開后,相當于選 kkk 個自然數和為 nnn 的方案數:先給 kkk 個數加 111 轉化問題為 kkk 個正整數和為 n+kn+kn+k,運用插板法得知方案數為 (n+k?1k?1)\dbinom{n+k-1}{k-1}(k?1n+k?1?)。
回到這一題 1(1?x)4\frac{1}{(1-x)^4}(1?x)41? 的第 n?1n-1n?1 項系數為 (n?23)\dbinom{n-2}{3}(3n?2?),乘回 xxx 后即為答案。代碼略。
Polya 計數
等我完全理解完再回來補。先給結論:用 kkk 中顏色給 nnn 個點的環染色方案數為 1n∑i=0n?1kgcd?(n,i)\dfrac{1}{n}\displaystyle\sum_{i=0}^{n-1}k^{\gcd(n,i)}n1?i=0∑n?1?kgcd(n,i),可否感性理解呢?
洛谷 P4980 【模板】Pólya 定理
題意
用 nnn 種顏色給 nnn 個點染色的方案數。兩種方案相同,當且僅當一種方案能夠旋轉變為“另一方案”。
n≤109n\le 10^9n≤109。
思路
問題是如何快速算出 1n∑i=0n?1ngcd?(n,i)\dfrac{1}{n}\displaystyle\sum_{i=0}^{n-1}n^{\gcd(n,i)}n1?i=0∑n?1?ngcd(n,i)。不妨暴力枚舉 nnn 的約數 d=gcd?(n,i)d=\gcd(n,i)d=gcd(n,i),變為:
1n∑d∣nndφ(nd)\dfrac{1}{n}\displaystyle\sum_{d|n}n^d\varphi(\frac{n}{d})n1?d∣n∑?ndφ(dn?)
nnn 太大了不能預處理 φ\varphiφ,記得勤取模。時間復雜度為 T(n)=∑i≥0{O(i)+O(n/i)}=O(n3/4)T(n)=\displaystyle\sum_{i\ge 0}\{O(\sqrt{i})+O(\sqrt{n/i})\}=O(n^{3/4})T(n)=i≥0∑?{O(i?)+O(n/i?)}=O(n3/4)(根據主定理推得)。代碼略。
第二類斯特林數
為什么先說第二類,因為第二類好像常用一些。
斯特林數將兩個典型的模型的方案數求解變成一個類似組合數的東西。
概念
{nm}\begin{Bmatrix} n\\ m \end{Bmatrix}{nm?} 表示,將 nnn 個不同的球放進 mmm 個相同的盒子里,且每個盒子都非空的方案數。
其遞推式的推導:
- 球 nnn 單開一個盒子,那么剩下 n?1n-1n?1 個球放進 m?1m-1m?1 個盒子:{n?1m?1}\begin{Bmatrix} n-1\\ m-1 \end{Bmatrix}{n?1m?1?};
- 球 nnn 放進非空的 mmm 個盒子:m{n?1m}m\begin{Bmatrix} n-1\\ m \end{Bmatrix}m{n?1m?}。
所以 {nm}={n?1m?1}+m{n?1m}\begin{Bmatrix} n\\ m \end{Bmatrix}=\begin{Bmatrix} n-1\\ m-1 \end{Bmatrix}+m\begin{Bmatrix} n-1\\ m \end{Bmatrix}{nm?}={n?1m?1?}+m{n?1m?}。
特別的,{n0}=[n=0]\begin{Bmatrix} n\\ 0 \end{Bmatrix}=[n=0]{n0?}=[n=0],中括號表示艾弗森括號。
可以 O(n2)O(n^2)O(n2) 預處理。
應用
先證明其通項公式,發現其滿足下降冪性質:
CF932E Team Work
題意
給定 n,kn,kn,k,求 ∑i=1n(ni)ik\displaystyle\sum_{i=1}^n\dbinom{n}{i}i^ki=1∑n?(in?)ik。
n≤109n\le 10^9n≤109,k≤5000k\le 5000k≤5000。
思路
把組合數用階乘拆開,再運用斯特林數下降冪的性質:
∑i=0nn!i!(n?i)!∑j=0k{kj}i!(i?j)!\sum_{i=0}^n\frac{n!}{i!(n-i)!}\sum_{j=0}^k\begin{Bmatrix} k\\ j \end{Bmatrix}\frac{i!}{(i-j)!}i=0∑n?i!(n?i)!n!?j=0∑k?{kj?}(i?j)!i!?
改變 iii 和 jjj 的枚舉順序,iii 從 jjj 開始到 nnn:
∑j=0k{kj}∑i=jnn!(n?i)!(i?j)!\sum_{j=0}^k\begin{Bmatrix} k\\ j \end{Bmatrix}\sum_{i=j}^n\frac{n!}{(n-i)!(i-j)!}j=0∑k?{kj?}i=j∑n?(n?i)!(i?j)!n!?
強制變成組合數:
∑j=0k{kj}n!(n?j)!∑i=jn(n?j)!(n?i)!(i?j)!\sum_{j=0}^k\begin{Bmatrix} k\\ j \end{Bmatrix}\frac{n!}{(n-j)!}\sum_{i=j}^n\frac{(n-j)!}{(n-i)!(i-j)!}j=0∑k?{kj?}(n?j)!n!?i=j∑n?(n?i)!(i?j)!(n?j)!?
∑j=0k{kj}n!(n?j)!∑i=jn(n?ji?j)\sum_{j=0}^k\begin{Bmatrix} k\\ j \end{Bmatrix}\frac{n!}{(n-j)!}\sum_{i=j}^n\binom{n-j}{i-j}j=0∑k?{kj?}(n?j)!n!?i=j∑n?(i?jn?j?)
化成 222 次冪形式:
∑j=0k{nm}n!(n?j)!∑i=0n?j(n?ji)\sum_{j=0}^k\begin{Bmatrix} n\\ m \end{Bmatrix}\frac{n!}{(n-j)!}\sum_{i=0}^{n-j}\binom{n-j}{i}j=0∑k?{nm?}(n?j)!n!?i=0∑n?j?(in?j?)
∑j=0k{kj}n!(n?j)!×2n?j\sum_{j=0}^k\begin{Bmatrix} k\\ j \end{Bmatrix}\frac{n!}{(n-j)!}\times2^{n-j}j=0∑k?{kj?}(n?j)!n!?×2n?j
O(k2)O(k^2)O(k2) 時間復雜度預處理第二類斯特林數即可。
代碼
#include<bits/stdc++.h>
using namespace std;
#define ll long long
const ll N=5002,mod=1e9+7;
ll n,k;
ll S2[N][N],p2[N];
ll qpow(ll x,ll k)
{ll ret=1;while(k){if(k&1)ret=ret*x%mod;x=x*x%mod;k>>=1;}return ret;
}
void init()
{S2[0][0]=1;for(int i=1;i<N;i++)for(int j=1;j<N;j++)S2[i][j]=(S2[i-1][j-1]+j*S2[i-1][j]%mod)%mod;
}
int main()
{scanf("%lld%lld",&n,&k);init();ll ans=0;for(int j=0;j<=min(n,k);j++){ll ret=S2[k][j]*qpow(2,n-j)%mod,mul=1;for(int t=n-j+1;t<=n;t++)ret=ret*t%mod;ans=(ans+ret)%mod;}printf("%lld",ans);return 0;
}
第一類斯特林數
概念
[nm]\begin{bmatrix} n\\ m \end{bmatrix}[nm?] 表示,將 nnn 個不同的球擺成 mmm 個環的方案數。兩個環相同,當且僅當一個環通過旋轉可以變成第二個環。
其遞推式的推導:
- 球 nnn 單開一個環,那么剩下 n?1n-1n?1 個球成 m?1m-1m?1 個環:[n?1m?1]\begin{bmatrix} n-1\\ m-1 \end{bmatrix}[n?1m?1?];
- 球 nnn 放進 mmm 個環,在放了的 (n?1)(n-1)(n?1) 個球后面都可以放:(n?1)[n?1m](n-1)\begin{bmatrix} n-1\\ m \end{bmatrix}(n?1)[n?1m?]。
所以 [nm]=[n?1m?1]+(n?1)[n?1m]\begin{bmatrix} n\\ m \end{bmatrix}=\begin{bmatrix} n-1\\ m-1 \end{bmatrix}+(n-1)\begin{bmatrix} n-1\\ m \end{bmatrix}[nm?]=[n?1m?1?]+(n?1)[n?1m?]。
特別的,[nm]=[n=0]\begin{bmatrix} n\\ m \end{bmatrix}=[n=0][nm?]=[n=0],中括號為艾弗森括號。
洛谷 P4609 FJOI2016 建筑師
題意
小 Z 是一個很有名的建筑師,有一天他接到了一個很奇怪的任務:在數軸上建 nnn 個建筑,每個建筑的高度是 111 到 nnn 之間的一個整數。
小 Z 有很嚴重的強迫癥,他不喜歡有兩個建筑的高度相同。另外小 Z 覺得如果從最左邊(所有建筑都在右邊)看能看到 AAA 個建筑,從最右邊(所有建筑都在左邊)看能看到 BBB 個建筑,這樣的建筑群有著獨特的美感。現在,小 Z 想知道滿足上述所有條件的建筑方案有多少種?
如果建筑 iii 的左(右)邊沒有任何建造比它高,則建筑 iii 可以從左(右)邊看到。兩種方案不同,當且僅當存在某個建筑在兩種方案下的高度不同。
1≤n≤50000,1≤A,B≤100,1≤T≤2000001 \leq n \leq 50000, \ 1 \leq A, B \leq 100, \ 1 \leq T \leq 2000001≤n≤50000,?1≤A,B≤100,?1≤T≤200000。
思路
中間那個左右都能看見的,高度必然是極值 nnn。
除去中間的,左邊要看到 a?1a-1a?1 個,右邊要看到 b?1b-1b?1 個。不妨將能看到的和其左/右邊比它矮的看成 a?1+b?1=a+b?2a-1+b-1=a+b-2a?1+b?1=a+b?2 個環(反正互換位置之后,盡管某些環的元素種類改變,但是環的個數不改變),就可以用第一類斯特林數的含義,將剩下 n?1n-1n?1 和分成 a+b?2a+b-2a+b?2 個環,即 [n?1a+b?2]\begin{bmatrix} n-1\\ a+b-2 \end{bmatrix}[n?1a+b?2?]。
然后再 a+b?2a+b-2a+b?2 個環中選各自最大值 a?1a-1a?1 個,成為左邊能看見的(排列方式唯一,必然是從矮到高),即 (a+b?2a?1)\dbinom{a+b-2}{a-1}(a?1a+b?2?)。
于是答案為 [n?1a+b?2](a+b?2a?1)\begin{bmatrix} n-1\\ a+b-2 \end{bmatrix}\dbinom{a+b-2}{a-1}[n?1a+b?2?](a?1a+b?2?)。
預處理第一類斯特林數即可。
代碼
#include<bits/stdc++.h>
using namespace std;
#define ll long long
const ll N=5e4+9,M=202,mod=1e9+7;
ll T;
ll S1[N][M],fac[N],inv[N];
ll qpow(ll x,ll k)
{ll ret=1;while(k){if(k&1)ret=ret*x%mod;x=x*x%mod;k>>=1;}return ret;
}
void init()
{S1[0][0]=1;for(int i=1;i<N;i++)for(int j=1;j<M;j++)S1[i][j]=(S1[i-1][j-1]+(i-1)*S1[i-1][j]%mod)%mod;fac[0]=1;for(int i=1;i<N;i++)fac[i]=fac[i-1]*i%mod;inv[N-1]=qpow(fac[N-1],mod-2);for(int i=N-2;i>=0;i--)inv[i]=inv[i+1]*(i+1)%mod;
}
ll C(ll n,ll m)
{return fac[n]*inv[m]%mod*inv[n-m]%mod;
}
int main()
{scanf("%lld",&T);init();while(T--){ll n,a,b;scanf("%lld%lld%lld",&n,&a,&b);printf("%lld\n",S1[n-1][a+b-2]*C(a+b-2,a-1)%mod);}return 0;
}
無根樹的 prufer 序列
概念
prufer 序列將節點數為 nnn 的無根樹轉化為長為 n?2n-2n?2 的序列。其表示期望與根的編號無關——定義為每次選擇一個編號最小的葉節點并刪除,然后在序列中記錄其所連接的節點編號,直到剩下兩個節點。
如上圖是一棵有 777 個節點的樹的 prufer 序列形成過程。
用 prufer 序列同樣可以還原出一棵樹來。
求法
求一棵樹的 prufer 序列。可以維護每個節點的度數,以節點編號為第一關鍵字扔進堆里排序,當度為 000 時舍棄這個節點。容易做到 O(nlog?n)O(n\log n)O(nlogn)。
有 O(n)O(n)O(n) 的更優秀的算法:“剝葉子”的過程中,每次只會讓其父親節點度數減 111,因此每刪除一個葉節點,至多產生一個新葉節點。
因此可以用一個單調指針 pospospos 維護當前度數為 000 的最小結點,pospospos 枚舉到了先刪 pospospos,然后看 faposfa_{pos}fapos? 是否度數變為 000,如果為 000 就順帶刪去了;沒準是條鏈,因此可以 while
一路刪上去。
用 prufer 還原一棵樹維護序列中節點的待加邊,當兒子連完、待加邊歸 000,就連向序列中下一個節點——似乎完全就是求 prufer 的相反操作,具體看下面的代碼。
P6086 【模板】Prufer 序列
思路
如上所述。
代碼
#include<bits/stdc++.h>
using namespace std;
#define ll long long
const ll N=1e5+9;
ll n,m,mod,k;
ll fa[N],siz[N];
ll fz(ll x)
{while(x!=fa[x])x=fa[x]=fa[fa[x]];return x;
}
void join(ll u,ll v)
{ll fu=fz(u),fv=fz(v);if(fu==fv)return;fa[fu]=fv;siz[fv]+=siz[fu];k--;
}
ll qpow(ll x,ll k)
{ll ret=1;while(k){if(k&1)ret=ret*x%mod;x=x*x%mod;k>>=1;}return ret;
}
int main()
{scanf("%lld%lld%lld",&n,&m,&mod);if(mod==1){puts("0");return 0;}k=n;for(int i=1;i<=n;i++)fa[i]=i,siz[i]=1;for(int i=1;i<=m;i++){ll u,v;scanf("%lld%lld",&u,&v);join(u,v);}if(k==1){puts("1");return 0;}ll ans=1;for(int i=1;i<=n;i++)if(i==fz(i))ans=ans*siz[i]%mod;ans=ans*qpow(n,k-2)%mod;printf("%lld",ans);return 0;
}
prufer 序列怎么求其實應用不廣,關鍵是其本身所帶有的性質更為重要。
應用
-
把一個樹化為 prufer 序列,是建立 nnn 個節點無根樹和 n?2n-2n?2 個元素的序列的雙射。
-
一個節點在 prufer 序列中出現次數為其度數減 111;
-
Cayley 公式:對 nnn 個節點建立一棵無根樹,有 nn?2n^{n-2}nn?2 種方案。
證明:任意一個長度為 n?2n-2n?2 的值域 [1,n][1,n][1,n] 的整數序列都可以通過 prufer 序列雙射對應一個生成樹,于是方案數就是 nn?2n^{n-2}nn?2。
CF156D Clues
題意
給定一個 nnn 個點 mmm 條邊的帶標號無向圖,若其有 kkk 個連通塊,想要添加 k?1k-1k?1 條邊使得整個圖連通。求方案數。
1≤n≤1051\le n\le 10^51≤n≤105,0≤m≤1050\le m\le 10^50≤m≤105,1≤k≤1091\le k\le 10^91≤k≤109。
思路
這是一道經典問題。
設每個連通塊有 sis_isi? 個節點,did_idi? 為連通塊的度數—— prufer 序列就是用度數所推。
計算上式即可。代碼略。
洛谷 P2290 HNOI2004 樹的計數
題意
一個有 nnn 個節點的樹,設它的節點分別為 v1,v2,…,vnv_1,v_2,\ldots,v_nv1?,v2?,…,vn?,已知第 iii 個節點 viv_ivi? 的度數為 did_idi?,問滿足這樣的條件的不同的樹有多少棵。
1≤n≤1501\le n\le 1501≤n≤150,保證滿足條件的樹不超過 101710^{17}1017 個。
思路
其實方案數就是上一篇題解所提,(n?2d1?1,d2?1,d3?1,......,dn?1)=(n?2)!∏i=1n(di?1)!\dbinom{n-2}{d_1-1,d_2-1,d_3-1,......,d_n-1}=\dfrac{(n-2)!}{\displaystyle\prod_{i=1}^n(d_i-1)!}(d1??1,d2??1,d3??1,......,dn??1n?2?)=i=1∏n?(di??1)!(n?2)!?。
好像會爆 long long
,階乘的增長速度超乎想象,倒不如用它的普通組合數意義形式。設 si=di?1s_i=d_i-1si?=di??1,sum=n?2sum=n-2sum=n?2(理論上是 ∑si\sum s_i∑si?,如果不是就不合法)。答案就是 ∏i=1n(sum?∑j=1i?1sjsi)\prod_{i=1}^n\binom{sum-\displaystyle\sum_{j=1}^{i-1}s_j}{s_i}i=1∏n?(si?sum?j=1∑i?1?sj??)
代碼略。
洛谷 P2624 HNOI2008 明明的煩惱
題意
給出標號為 111 到 nnn 的點,以及某些點最終的度數,允許在任意兩點間連線,可產生多少棵度數滿足要求的樹?無解就輸出 ?1-1?1。
1≤n≤10001\le n\le10001≤n≤1000。
思路
這是上一題的加強版。
對于已知的 mmm 個點,我們可以直接套用上面的 (m?2d1?1,d2?1,d3?1,......,dm?1)=(m?2)!∏i=1m(di?1)!\dbinom{m-2}{d_1-1,d_2-1,d_3-1,......,d_m-1}=\dfrac{(m-2)!}{\displaystyle\prod_{i=1}^m(d_i-1)!}(d1??1,d2??1,d3??1,......,dm??1m?2?)=i=1∏m?(di??1)!(m?2)!? 公式,prufer 序列上已經選了的方案數為 (n?2∑i=1m(di?1))\dbinom{n-2}{\displaystyle\sum_{i=1}^m(d_i-1)}(i=1∑m?(di??1)n?2?),最后再把剩下的 n?mn-mn?m 個點,塞進 n?2?∑i=1m(di?1)n-2-\displaystyle\sum_{i=1}^m(d_i-1)n?2?i=1∑m?(di??1) 去:
(n?2∑i=1m(di?1))(m?2)!∏i=1m(di?1)!×(n?m)n?2?∑i=1m(di?1)\dbinom{n-2}{\displaystyle\sum_{i=1}^m(d_i-1)}\dfrac{(m-2)!}{\displaystyle\prod_{i=1}^m(d_i-1)!}\times(n-m)^{n-2-\displaystyle\sum_{i=1}^m(d_i-1)}(i=1∑m?(di??1)n?2?)i=1∏m?(di??1)!(m?2)!?×(n?m)n?2?i=1∑m?(di??1)
沒得化回普通組合數,要用高精捏。代碼略,有緣再寫。