Password Attacker

题意就是给 M 个关键字,组合成 N 字符长度的结果,每一个关键字都必须在 N 位的字符中出现,有多少种可能结果。

范围 1 ≤ M ≤ N ≤ 100.

举例假设 M = 3(key = 3, 7, 5)  N = 4 位字符长度

结果可以为3577, 3557, 7353, 5735.

下面的结果是不合法的

1357 // 1 没有在key中,不是合法关键字

3355 // 7 是关键字,但是结果中没有出现

357 // 结果必须是一个4位长度

思路1:

设立状态dp(i, j)表示在 M 个关键字中取前 i 个不同关键字,组成长度为 j . 所有结果总和

划分子问题

1. i == j 时候, 取前 i 个Key组成长度为 i ,结果为

2. i < j 时候,考虑dp(i-1, k),选去第 i 个元素j-k次,第i个元素与前i-1个元素不同,把这j-k个第i个元素插入到已有长度k的序列中,就可以转移到dp(i, j)

3. i > j, 不考虑

那么对于2,如何插入?

已有的 k 个元素有 k+1 个位置可以进行插入,问题就是把 j-k 个元素插入到 k+1 个桶中有多少种结果?

表示第 i 个桶,那么问题就等价于的所有非负整数解。

我们把两边同时加上k+1,变成 的所有正整数解。

相当于给j+1个1,插入隔板把它们分到k+1个桶中,每个桶要保证至少有一个元素,所以一共有 j 个可以提供插入隔板的位置,总共有k+1个桶,所以需要在这 j 个空隙中选择出k个位置插入隔板就成k+1个桶。

答案为

最后的2,也就等于 

最后的答案就是dp(m, n)

源码如下:

#include <bits/stdc++.h>
using namespace std; typedef long long int64; const int64 M = 1000000007LL;
const int maxn = 105;
int64 C[maxn][maxn];
int64 dp[maxn][maxn]; void getCombination() {
int n = maxn;
for (int i = 1; i <= n; i++)
C[i][0] = C[i][i] = 1LL; for (int i = 2; i <= n; i++)
for (int j = 1; j <= i; j++)
C[i][j] = (C[i-1][j-1] % M + C[i-1][j] % M) % M;
} int main() { getCombination();
int t, m, n, cas = 1;
scanf("%d", &t);
while ( t-- ) {
scanf("%d%d", &m, &n);
// init
memset(dp, 0LL, sizeof(dp));
for (int i = 1; i <= n; i++) dp[1][i] = 1;
for (int i = 2; i <= n; i++) dp[i][i] = (dp[i-1][i-1] * (int64)i) % M; for (int i = 2; i <= m; i++)
for (int j = i+1; j <= n; j++)
for (int k = i-1; k <= j-1; k++)
dp[i][j] = (dp[i][j] + (C[j][k] * dp[i-1][k]) % M) % M; printf("Case #%d: %lld\n", cas++, dp[m][n]);
}
return 0;
}

思路2:

qqz大神,dp(i, j)表示从 M 个元素中随机选取 i 个不同的元素组成 j 的长度的所有结果

划分子问题

1. i == j, dp(i, i) 随机从 M 个元素选取 i 个元素的全排列,为

2. i == 1, dp(1, j) 表示从 M 个元素中随机选取一个重复 j 次,结果为 M

3. i < j , ,

和的第一部分表示在dp(i-1, j-1)基础上,从剩下的M-i+1个数中取1个放到最后组成dp(i, j). 和的第二部分表示从已经选出的i个元素中挑1个放到最后。

最后的答案就是dp(m, n)

源码如下:

#include <bits/stdc++.h>
using namespace std; using int64 = long long;
const int64 M = 1000000007LL; int64 dp[105][105], A[105][105]; void initA() { for (int i = 1; i <= 100; i++) {
A[i][0] = 1;
for (int j = 1; j <= i; j++)
A[i][j] = ((i-j+1) * A[i][j-1]) % M;
}
} int m, n; int64 f(int i, int j) {
if (dp[i][j] != -1) return dp[i][j];
if (i == j) return dp[i][j] = A[m][i];
else return dp[i][j] = (((m-i+1) * f(i-1, j-1)) % M + (i * f(i, j-1)) % M ) % M;
} int main() {
freopen("A-large-practice.in", "r", stdin);
freopen("a_out.txt", "w", stdout);
int t, cas = 1;
initA();
scanf("%d", &t);
while (t--) {
scanf("%d%d", &m, &n);
for (int i = 1; i <= m; i++)
for (int j = 1; j <= n; j++)
dp[i][j] = -1;
for (int i = 1; i <= n; i++) dp[1][i] = m;
printf("Case #%d: %lld\n", cas++, f(m ,n));
}
return 0;
}

【DP】组合数字的更多相关文章

  1. n全排列输出和 n个数的组合(数字范围a~b)

    n全排列输出: int WPermutation(int num, bool bRepeat) num表示num全排列 bRepeat标志是否产生重复元素的序列. int Permutation(in ...

  2. hdu 4945 2048 (dp+组合的数目)

    2048 Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total Submi ...

  3. #DP# ----- OpenJudge数字组合

    OpenJudge 2985:数字组合 总时间限制:1000ms  内存限制: 65536kB 描述 有n个正整数,找出其中和为t(t也是正整数)的可能的组合方式.如:n=5,5个数分别为1,2,3, ...

  4. ZOJ-3380 Patchouli’s Spell Cards DP, 组合计数

    题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3380 题意:有m种不同的元素,每种元素都有n种不同的相位,现在假 ...

  5. 3.29省选模拟赛 除法与取模 dp+组合计数

    LINK:除法与取模 鬼题.不过50分很好写.考虑不带除法的时候 其实是一个dp的组合计数. 考虑带除法的时候需要状压一下除法操作. 因为除法操作是不受x的大小影响的 所以要状压这个除法操作. 直接采 ...

  6. CF 258B Little Elephant and Elections [dp+组合]

    给出1,2,3...m 任取7个互不同样的数a1,a2,a3,a4,a5,a6,a7 一个数的幸运度是数位上4或7的个数 比方244.470幸运度是2. 44434,7276727.4747,7474 ...

  7. bzoj3622已经没有什么好害怕的了 dp+组合+容斥(?)

    3622: 已经没有什么好害怕的了 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1033  Solved: 480[Submit][Status][ ...

  8. 数位dp——奏响数字数位的美妙乐章

    数位dp:处理数字数位关系的一种dp方式. 一般的题目特征十分明显: 1.一般和数字本身有很大关系. 2.一般求数字在区间L,R中的一些信息 3.L,R一般很大,通常能达到long long级别. d ...

  9. HihoCoder 1075 开锁魔法III(概率DP+组合)

    描述 一日,崔克茜来到小马镇表演魔法. 其中有一个节目是开锁咒:舞台上有 n 个盒子,每个盒子中有一把钥匙,对于每个盒子而言有且仅有一把钥匙能打开它.初始时,崔克茜将会随机地选择 k 个盒子用魔法将它 ...

随机推荐

  1. jsp 内置对象

    1.Request 指属性在一次请求范围内有效.如果页面从给一个页面跳转到另一个页面,那么该属性就失效了.这里所指的跳转是指客户端跳转,比如客户单击超链接跳转到其他页面或者通过浏览器地址栏浏览其他页面 ...

  2. man ascii

    Linux 2.6 - man page for ascii (linux section 7) - Unix & Linux Commands Linux 2.6 - man page fo ...

  3. ASP.NET ZERO 学习 HangFire的使用二

    之前在ABP配置好了HangFire环境之后,那么如何才能添加Job到HangFire中让其执行呢 这就需要参考ABP的Background Jobs and Workers, 参考路径:http:/ ...

  4. maven css/js 压缩配置

    <plugin>               <groupId>net.alchim31.maven</groupId>               <art ...

  5. flash中htmlText样式

    赋值htmlText时,htmlText中指定了样式的按指定样式显示,没有指定的按该文本的默认样式显示. PS:flash软件拖出来的文本,在赋值htmlText的时候该文本原有样式会失效,而new出 ...

  6. Spring事务管理(转)

    1 初步理解 理解事务之前,先讲一个你日常生活中最常干的事:取钱. 比如你去ATM机取1000块钱,大体有两个步骤:首先输入密码金额,银行卡扣掉1000元钱:然后ATM出1000元钱.这两个步骤必须是 ...

  7. Android消息的提示,Toast吐司方式

    1:选中某个控件进行触发 2:触发事件进行监听,然后绑定Toast对象进行消息提示 1,创建Android项目的时候,自带的一个Activity,我们看看代码 package com.example. ...

  8. Hishop网站迁移后出现DataProtectionConfigurationProvider错误

    错误代码如下: 配置错误 说明: 在处理向该请求提供服务所需的配置文件时出错.请检查下面的特定错误详细信息并适当地修改配置文件. 分析器错误信息: 未能使用提供程序“DataProtectionCon ...

  9. 【转】如何理解c和c++的复杂类型声明

    转自:http://blog.chinaunix.net/space.php?uid=22889411&do=blog&id=59667 曾经碰到过让你迷惑不解.类似于int * (* ...

  10. package、import和import static

    package 语句: 该语句必须作为源文件的第一条非注释性语句,一个源文件只能指定一个包,即只能包含一条package语句. import 和import static 关键字: 引入import关 ...