[BZOJ2339][HNOI2011]卡农

时间:2021-11-03 23:39:31

[BZOJ2339][HNOI2011]卡农

试题描述

[BZOJ2339][HNOI2011]卡农

输入

见“试题描述

输出

见“试题描述

输入示例

见“试题描述

输出示例

见“试题描述

数据规模及约定

见“试题描述

题解

先考虑 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 优化读入。。。

[BZOJ2339][HNOI2011]卡农的更多相关文章

  1. bzoj2339&lbrack;HNOI2011&rsqb;卡农 dp&plus;容斥

    2339: [HNOI2011]卡农 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 842  Solved: 510[Submit][Status][ ...

  2. BZOJ2339&lbrack;HNOI2011&rsqb;卡农——递推&plus;组合数

    题目链接: [HNOI2011]卡农 题目要求从$S=\{1,2,3……n\}$中选出$m$个子集满足以下三个条件: 1.不能选空集 2.不能选相同的两个子集 3.每种元素出现次数必须为偶数次 我们考 ...

  3. BZOJ2339 HNOI2011卡农(动态规划&plus;组合数学)

    考虑有序选择各子集,最后除以m!即可.设f[i]为选i个子集的合法方案数. 对f[i]考虑容斥,先只满足所有元素出现次数为偶数.确定前i-1个子集后第i个子集是确定的,那么方案数为A(2n-1,i-1 ...

  4. 【BZOJ2339】&lbrack;HNOI2011&rsqb;卡农 组合数&plus;容斥

    [BZOJ2339][HNOI2011]卡农 题解:虽然集合具有无序性,但是为了方便,我们先考虑有序的情况,最后将答案除以m!即可. 考虑DP.如果我们已经知道了前m-1个集合,那么第m个集合已经是确 ...

  5. 【BZOJ2339】卡农(递推,容斥)

    [BZOJ2339]卡农(递推,容斥) 题面 BZOJ 题解 先简化一下题意: 在\([1,2^n-1]\)中选择不重复的\(m\)个数,使得他们异或和为\(0\)的方案数. 我们设\(f[i]\)表 ...

  6. P3214 &lbrack;HNOI2011&rsqb;卡农

    题目 P3214 [HNOI2011]卡农 在被一题容斥\(dp\)完虐之后,打算做一做集合容斥这类的题了 第一次深感HNOI的毒瘤(题做得太少了!!) 做法 求\([1,n]\)组成的集合中选\(m ...

  7. &lbrack;HNOI2011&rsqb;卡农

    题目描述 众所周知卡农是一种复调音乐的写作技法,小余在听卡农音乐时灵感大发,发明了一种新的音乐谱写规则.他将声音分成 n 个音阶,并将音乐分成若干个片段.音乐的每个片段都是由 1 到 n 个音阶构成的 ...

  8. &lbrack;HNOI2011&rsqb;卡农 题解

    题目描述 众所周知卡农是一种复调音乐的写作技法,小余在听卡农音乐时灵感大发,发明了一种新的音乐谱写规则.他将声音分成 n 个音阶,并将音乐分成若干个片段.音乐的每个片段都是由 1 到 n 个音阶构成的 ...

  9. 【bzoj2339】&lbrack;HNOI2011&rsqb;卡农 dp&plus;容斥原理

    题目描述 题解 dp+容斥原理 先考虑有序数列的个数,然后除以$m!$即为集合的个数. 设$f[i]$表示选出$i$个集合作为满足条件的有序数列的方案数. 直接求$f[i]$较为困难,考虑容斥,满足条 ...

随机推荐

  1. cs程序添加初始化加载

    this.Name = "mysirst"; this.Text = "车辆窗体程序"; this.Load += new System.EventHandle ...

  2. NOSQL 数据库 CodernityDB

    CodernityDB 是一个开源的纯 Python 实现的.无第三方依赖.支持多平台的 NoSQL 数据库. 关键特性: 纯 Python 开发 支持多索引 快速 (每秒将近10万的写入和超过10万 ...

  3. Mybatis批量插入oracle&comma;mysql

    oracle  <insert id="addUserData" parameterType="java.util.List"> INSERT IN ...

  4. iOS几种简单有效的数组排序方法

    第一种,利用数组的sortedArrayUsingComparator调用 NSComparator ,obj1和obj2指的数组中的对象 NSComparator cmptr = ^(id obj1 ...

  5. 11&period;2&period;0&period;3&period;7 PSU补丁升级

    说明:这是新上线主机,还没有建库的.如果库已经存在,那么最后还要进行刷库操作呢! 环境:SUSE Linux Enterprise Server 11 sp1 (x86_64) 要求将数据库从11.2 ...

  6. javascript 函数的基础知识

    1. Why JavaScript functions always return a value? I'm taking a course in JavaScript programming, an ...

  7. shiro自定义realm认证&lpar;五&rpar;

    上一节介绍了realm的作用: realm:需要根据token中的身份信息去查询数据库(入门程序使用ini配置文件),如果查到用户返回认证信息,如果查询不到返回null.token就相当于是对用户输入 ...

  8. ORACLE常用数值函数、转换函数、字符串函数介绍

    ORACLE常用数值函数.转换函数.字符串函数介绍. 数值函数: abs(m) m的绝对值 mod(m,n) m被n除后的余数 power(m,n) m的n次方 round(m[,n]) m四舍五入至 ...

  9. json转对象

    1,引入依赖 <dependency> <groupId>net.sf.json-lib</groupId> <artifactId>json-lib& ...

  10. 一个简单的Javascript闭包示例

    //=====用闭包实现函数的Curry化===== //数字求和函数的函数生成器 function addGenerator( num ){ //返回一个简单的匿名函数,求两个数的和,其中第一个数字 ...