NYOJ 832合并游戏(状态压缩dp)

描述
大家都知道Yougth除了热爱编程之外,他还有一个爱好就是喜欢玩。
某天在河边玩耍的时候,他发现了一种神奇的石子,当把两个石子放在一起的时候,后一个石子会消失,而且会蹦出一定数量的金币,这可乐坏了Yougth,但是他想得到最多的金币,他该怎么做?
输入
首先一行,一个n(1<=n<=10),表示有n个石子。
接下来n*n的一个矩阵,Aij表示第i个和第j个合并蹦出的金币值(小于10000,注意合并后j会消失)。
输出
输出最多能得到的金币值。
样例输入
2
0 4
1 0
3
0 20 1
12 0 1
1 10 0
样例输出
4

22


这一题典型的状态压缩dp, 直接解释状态压缩方程。这题状态压缩方程不好用式子表示,我大概举个例子讲一下怎么转移的。首先要来一个数来表示目前的一个状态,比如n==8,就是有8个石子,然后接着就是假设有一个状态10101101(十进制173),0表示石子已经没掉了,1表示这个石子还在,那么这个状态其实可以由11101101或者10111101或者10101111这三个状态转化而来,而且就这三个, 而这三个状态相比于10101101这个数多出来的1就是相当于与10101101中的1表示的石子合并时没掉的。这个1应该是变成10101101中的0,所以每找到一个0,就枚举这个状态中所有1,因为与可能是这个1让原来的1没掉了,变成了零。以上举了个例子应该就清楚大概思路了清楚了。


AC代码:


# include <cstdio>
# include <cstring>
# include <algorithm>
# include <vector>
using namespace std;
vector<int> v[20];
int bit[20], a[20][20], dp[15000];
int n;
int cal(int temp){
	int cnt=0;
	for(int i=1; i<=n; i++){
		if(!(temp&1)){
			cnt++;
		}
		temp=(temp>>1);
	}
	return cnt;
}
int main(){
	int i, j, k, temp1, l, ans;
	bit[1]=1;
	for(i=2; i<=10; i++){
		bit[i]=(bit[i-1]<<1);
	}
	while(scanf("%d", &n)!=EOF){
		if(n==1){
			scanf("%d", &temp1);
			printf("0\n");
			continue;
		}
		memset(dp, 0, sizeof(dp));
		dp[(1<<n)-1]=0;
		for(i=0; i<=10; i++){
			v[i].clear();
		}
		for(i=1; i<=n; i++){
			for(j=1; j<=n; j++){
				scanf("%d", &a[i][j]);
			}
		}
		for(i=1; i<=(1<<n)-1; i++){
			v[cal(i)].push_back(i);
		}
		for(i=1; i<=n-1; i++){
			for(j=0; j<=v[i].size()-1; j++){
				temp1=v[i][j];
				for(k=1; k<=n; k++){
					if(!(temp1&1)){
						for(l=1; l<=n; l++){
							if(v[i][j]&bit[l]){
								dp[v[i][j]]=max(dp[v[i][j]], dp[(v[i][j]|bit[k])]+a[l][k]);
							}
						}
					}
					temp1=(temp1>>1);
				}
			}
		}
		ans=0;
		for(i=0; i<=v[n-1].size()-1; i++){
			ans=max(ans, dp[v[n-1][i]]);
		}
		printf("%d\n", ans);
	}
	return 0;
}


你可能感兴趣的:(算法,动态规划,ACM,状态压缩,数位dp)