递推 ACdream1420 High Speed Trains

传送门:点击打开链接

题意: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)