dp02
dp02
今天刷了一道dp问题
题目描述
小明过生日的时候,爸爸送给他一副乌龟棋当作礼物。
乌龟棋的棋盘是一行 N 个格子,每个格子上一个分数(非负整数)。棋盘第 1格是唯一的起点,第 N格是终点,游戏要求玩家控制一个乌龟棋子从起点出发走到终点。
乌龟棋中 M张爬行卡片,分成 4 种不同的类型(M 张卡片中不一定包含所有 4 种类型的卡片,见样例),每种类型的卡片上分别标有
1,2,3,4 四个数字之一,表示使用这种卡片后,乌龟棋子将向前爬行相应的格子数。游戏中,玩家每次需要从所有的爬行卡片中选择一张之前没有使用过的爬行卡片,控制乌龟棋子前进相应的格子数,每张卡片只能使用一次。
游戏中,乌龟棋子自动获得起点格子的分数,并且在后续的爬行中每到达一个格子,就得到该格子相应的分数。玩家最终游戏得分就是乌龟棋子从起点到终点过程中到过的所有格子的分数总和。
很明显,用不同的爬行卡片使用顺序会使得最终游戏的得分不同,小明想要找到一种卡片使用顺序使得最终游戏得分最多。
现在,告诉你棋盘上每个格子的分数和所有的爬行卡片,你能告诉小明,他最多能得到多少分吗?
代码
#include<iostream>
#include<cstring>
#include<string>
#include<algorithm>
#include<queue>
#include<map>
#include<unordered_map>
#include<stack>
#include<vector>
typedef long long LL;
using namespace std;
#define INF 0x3f3f3f3f
typedef pair<int,int>PII;
#define x first
#define y second
const int N=360,M=130;
int dp[2][42][42][42][42];
int A[N],B[N],st[N];
int num[5];
int n,m,k;
int main(){
int n,m;
cin>>n>>m;
for(int i=1;i<=n;i++){
cin>>A[i];
}
for(int i=0;i<m;i++){
cin>>k;
num[k]++;
}
for(int i=1;i<=m;i++){
for(int j=0;j<=min(num[1],i);j++){
for(int k=0;k<=min(num[2],i-j);k++){
for(int u=0;u<=min(num[3],i-j-k);u++){
int v=i-j-k-u;
if(v>num[4])continue;
dp[i%2][j][k][u][v]=dp[(i-1)%2][j][k][u][v];
if(j!=0)
dp[i%2][j][k][u][v]=max(dp[i%2][j][k][u][v],dp[(i-1)%2][j-1][k][u][v]+A[j+k*2+u*3+4*v+1]);
if(k!=0)
dp[i%2][j][k][u][v]=max(dp[i%2][j][k][u][v],dp[(i-1)%2][j][k-1][u][v]+A[j+k*2+u*3+4*v+1]);
if(u!=0)
dp[i%2][j][k][u][v]=max(dp[i%2][j][k][u][v],dp[(i-1)%2][j][k][u-1][v]+A[j+k*2+u*3+4*v+1]);
if(v!=0)
dp[i%2][j][k][u][v]=max(dp[i%2][j][k][u][v],dp[(i-1)%2][j][k][u][v-1]+A[j+k*2+u*3+4*v+1]);
}
}
}
}
cout<<dp[m%2][num[1]][num[2]][num[3]][num[4]]+A[1];
return 0;
}