标签:
传送门:点击打开链接
题意:n个点,每个点至少被一条边覆盖,不存在自环和重边。问有多少种情况
一个非常有意思的递推,其实可以利用容斥的思想去考虑这道题目
设F[n]表示为n个点的情况数,那么最多会存在n*(n-1)/2条边,这些边中,每条边都是可有可无。如果不考虑不满足条件的情况(也就是有的点没有被边覆盖的情况),可能的情况数为pow(2,n*(n-1)/2),也就是考虑每一条边是否存在
那么对于答案而言,不能有点没有被覆盖,接下来我们就来枚举点被边覆盖的个数,设为k
当有k个点被覆盖了,这k个点的情况数就是F[k],那么这k个点分别是哪k个,需要组合数C(n,k),所以对于k个点被覆盖时,不满足条件的总类数就是F[k]*C(n,k)
k可以从1枚举到n-1,然后还要考虑一个特殊情况,就是没有一个点被边覆盖,这只有1种情况。
综上所述,答案是F[n]=pow(2,n*(n-1)/2)-segma(F[k]*C(n,k))其中1<=k<=n-1
import java.util.Scanner; import java.math.*; public class Main { static final int MX = 100 + 5; public static void main(String[] args) { Scanner cin = new Scanner(System.in); BigInteger zero = BigInteger.valueOf(0); BigInteger one = BigInteger.valueOf(1); BigInteger two = BigInteger.valueOf(2); BigInteger[][] C = new BigInteger[MX][MX]; C[0][0] = one; for (int i = 1; i < MX; i++) { C[i][0] = C[i][i] = one; for (int j = 1; j < i; j++) { C[i][j] = C[i - 1][j - 1].add(C[i - 1][j]); } } BigInteger f[] = new BigInteger[MX]; f[1] = one; f[2] = one; for (int i = 3; i < MX; i++) { f[i] = two.pow(i * (i - 1) / 2).subtract(one); for (int k = 2; k <= i - 1; k++) { f[i] = f[i].subtract(C[i][k].multiply(f[k])); } } while (cin.hasNext()) { int n = cin.nextInt(); System.out.println(f[n]); } } }
版权声明:本文为博主原创文章,未经博主允许不得转载。
递推 ACdream1420 High Speed Trains
标签:
原文地址:http://blog.csdn.net/qwb492859377/article/details/48027521