所有的代码都在这里 ,新队员们做不出来的题可以先参考我的代码,能自己讨论出来的尽量先多讨论(结合我的代码)
http://acm.hust.edu.cn/vjudge/contest/view.action?cid=29041#overview
我会一个个题更新。。。。
以前写过几个简单的dp入门提
http://blog.csdn.net/haha593572013/article/details/7834532
B - Marriage Ceremonies
学习资料
http://wenku.baidu.com/view/8c9bd904de80d4d8d15a4fd4.html
这道题纯暴力的复杂度是16!,肯定是没法过的。。。
可以想到这样的状态,当我们在决策第i行应该选哪一列的时候,我们需要知道前面i-1行已经选了哪些列了,对于这些列,我们并不关心到底是哪些行选择了这些列 ,我们关心的只是一个最大的权值和,即前i-1行已经选择了某个集合的列的前提下所能获取的最大权值和,然后我们枚举这个集合中还没有被选择的列在第i行进行状态转移。
那么接下来的事情就是怎么表示这个集合了。我们可以将这个集合压缩成一个整数,用这个整数的二进制表示来描述这个集合,然后我们可以发现我们可以用2^16个数来描述所有的状态了。
101011 = 43
上面这个状态代表的意义是第0 , 1 ,3 , 5 列都已经选择了。
我们就用 43这个数字来描述这个状态。
一些二进制的基本知识
判断j是否属于集合i:i&(1<
在集合i中去除j:i-(1<
在集合i中加入点j:i|(1<
#include
#include
const int maxn = 100010;
const int mod = 1000000007;
int dp[17][1<<17];
int a[17][17];
int main()
{
int n;
int t,ca=1;
scanf("%d",&t);
while(t--)
{
scanf("%d",&n);
for(int i = 0; i < n; i++)
{
for(int j = 0; j < n; j++)
{
scanf("%d",&a[i][j]);
}
}
for(int i = 0; i < n; i++)
{
for(int j = 0; j < two[n]; j++) dp[i][j] = -1;//一开始所有的状态都是非法的
}
for(int i = 0; i < n; i++) dp[0][1<
C - Love Calculator、
这个题本质其实就是LCS(longest comman sequence)的变形,想想看,最终构成的字符串的长度肯定是两个串的长度相加减去lcs的长度,但是有多少的这样的串呢?
初做dp的话还真是挺难想的,不过DP靠的就是多做多练了,感觉嘛,培养培养就有了,考虑这样的状态dp[i][j][k],表示构造了i长度的字符串,利用了第一个串的前i个字符以及第二个字符串的前j个字符,这个状态的意义就是当前状态下总的方案数,那么现在考虑转移,其实我们是要在后面继续扩充一个字符,如果a[j+1] = b[k+1],就可以转移到dp[i+1][j+1][k+1]的状态,
否则,可以转移到dp[i+1][j][k+1]或者dp[i+1][j+1][k];
求完DP后,我们需要寻找我们的答案,这个大家可以好好想想,应该不难。
同样的,还是贴上AC代码
注:代码一定要独立完成,切不可肆意模仿甚至抄袭。。。。
#include
#include
#include
#include
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
const int maxn = 100010;
const int mod = 1000000007;
long long dp[65][35][35];
char s1[110],s2[110];
int main()
{
int t,ca=1;
scanf("%d",&t);
while(t--)
{
scanf("%s%s",s1+1,s2+1);
int n = strlen(s1+1);
int m = strlen(s2+1);
s1[n+1] = '@'; s2[m+1]='@';
for(int i = 0; i <= m+n; i++)
{
for(int j = 0; j <= n; j++)
{
for(int k = 0; k <= m; k++)
{
dp[i][j][k] = 0;
}
}
}
dp[0][0][0] = 1;
for(int i = 0; i <= m+n; i++)
{
for(int j = 0; j <= n; j++)
{
for(int k = 0; k <= m; k++) if(dp[i][j][k])
{
if(s1[j+1]==s2[k+1])
{
dp[i+1][j+1][k+1] += dp[i][j][k];
}
if(s1[j+1]!=s2[k+1])
{
dp[i+1][j+1][k] += dp[i][j][k];
dp[i+1][j][k+1] += dp[i][j][k];
}
}
}
}
int ans_len = -1;
for(int i = 1; i <= m+n; i++)
{
if(dp[i][n][m])
{
ans_len = i;
break;
}
}
printf("Case %d: %d %lld\n",ca++,ans_len,dp[ans_len][n][m]);
}
return 0;
}