[BZOJ2339][HNOI2011]卡农

试题描述

[BZOJ2339][HNOI2011]卡农_i++

输入

见“试题描述

输出

见“试题描述

输入示例

见“试题描述

输出示例

见“试题描述

数据规模及约定

见“试题描述

题解

先考虑 m 个 01 串排顺序的情况。可以发现如果定下前 m - 1 个 01 串,那么第 m 个串就可以由前面所有 01 串按位异或得出,所以方案数为 A(2n - 1, m - 1)(即除全 0 串外的所有情况选择 m - 1 个并排列顺序的方案数),现在我们需要减去不合法的情况。我们令 f(m) 表示前 m 个串考虑顺序合法的方案数。

不合法#1:所有 m - 1 个 01 串的异或和为全 0 串,即最后一个填的是全 0 串,那么要减去 f(m - 1)。

不合法#2:最后一个填的串与前 m - 1 个串有重复,那么如果去掉这两个相同的串,剩下的串能组成合法的情况(即 f(m - 2)),然后这个串可能与前 m - 1 个串中任意一个串重复,所以要乘上 m - 1,最后这个重复的串本身有 2n - 1 - (m - 2) 种情况。所以最终需要减去 f(m - 2) * (m - 1) * [2n - 1 - (m - 2)]。

最后再把答案除以 m! 就好了,求一下逆元。

注意不可以用组合数的方法避免排顺序,因为如果这样做了,在“不合法#2”这一步中我们就不能保证对于那 m - 1 个重复的位置每一个都是完整的 f(m - 2) 中情况,可以想象一下如果用组合数,那么方案中的 01 串就按照一固定顺序排好了,那么对于一个特定的串它的位置就没有任意性了。

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <stack>
#include <vector>
#include <queue>
#include <cstring>
#include <string>
#include <map>
#include <set>
using namespace std;

int read() {
    int x = 0, f = 1; char c = getchar();
    while(!isdigit(c)){ if(c == '-') f = -1; c = getchar(); }
    while(isdigit(c)){ x = x * 10 + c - '0'; c = getchar(); }
    return x * f;
}

#define maxn 1000010
#define MOD 100000007
#define LL long long
int f[maxn], A[maxn];

void gcd(LL a, LL b, LL& x, LL& y) {
	if(!b){ x = 1; y = 0; return ; }
	gcd(b, a % b, y, x); y -= a / b * x;
	return ;
}
LL Inv(LL a) {
	LL x, y;
	gcd(a, MOD, x, y);
	x = (x % MOD + MOD) % MOD;
	return (x % MOD + MOD) % MOD;
}

int main() {
	int n = read(), m = read();
	
	int sum2 = 1;
	for(int i = 1; i <= n; i++) {
		sum2 <<= 1;
		if(sum2 >= MOD) sum2 -= MOD;
	}
	A[0] = 1;
	for(int i = 1; i <= m; i++) A[i] = (LL)A[i-1] * ((sum2 - i + MOD) % MOD) % MOD;
	f[1] = f[2] = 0; int tmp = 2;
	for(int i = 3; i <= m; i++) {
		f[i] = A[i-1] - ((LL)f[i-2] * (i - 1) % MOD) * ((sum2 - i + 1 + MOD) % MOD) % MOD;
		if(f[i] < 0) f[i] += MOD;
		f[i] -= f[i-1];
		if(f[i] < 0) f[i] += MOD;
		tmp = (LL)tmp * i % MOD;
	}
	printf("%d\n", (LL)f[m] * Inv(tmp) % MOD);
	
	return 0;
}

BTW,这题不能用 fread 优化读入。。。