本人是初出茅庐的小菜,代码繁琐,算法低端,写给自己看的,希望大神们多多指教
题目大意:
有一个人在两棵掉苹果的树下来回接苹果,每分钟都有个树往下掉,编号为1、2的两棵树,一共掉T分钟,然后这个人只能动W次,一开始站在1树,问最后最多能接多少个苹果。
思路:
令dp[x][y][z],为已经过去了x分钟,已经走了y步,目前在z(1 or 2)树下,并且苹果已经落下来了,因为初始在1树,所以令dp[0][0][1]=0,其他的都设为-1。
other idea:
可以把z省略,节省一维的空间,可以用y%2+1来表示所在的位置,这里懒得改了
#include <iostream>
#include <stdio.h>
#include <math.h>
#include <stdlib.h>
#include <string>
#include <string.h>
#include <algorithm>
#include <vector>
#include <queue>
#include <set>
#include <map>
#include <stack>
using namespace std;
typedef long long LL;
const int INF=0x7fffffff;
const int MAX_N=1009;
int T,W;
int A[MAX_N];
int dp[1009][35][3];
int change(int x){
if(x==1)x=2;
else x=1;
return x;
}
int main(){
cin>>T>>W;
for(int i=1;i<=T;i++){
scanf("%d",&A[i]);
}
memset(dp,-1,sizeof(dp));
dp[0][0][1]=0;
for(int i=1;i<=T;i++){
for(int j=0;j<=W;j++){
for(int k=1;k<=2;k++){
if(A[i]==k){//如果所在的树有苹果掉下,就把之前的两种状态(在这个树没动地方 or 从另一个树跑过来)取个最大值,然后再+1
if(j==0){//这一步是因为整个for的j是从0开始的,下面dp有j-1,所以我怕出现j-1=-1的情况,就加了两句,暂时不知道怎么处理好
dp[i][j][k]=dp[i-1][j][k]+1;
}
else dp[i][j][k]=1+max(dp[i-1][j][k],dp[i-1][j-1][change(k)]);
}
else{//如果所在的树没苹果,直接取前两种状态最大值,不用+1
if(j==0){//原理跟上面相同
dp[i][j][k]=dp[i-1][j][k];
}
else dp[i][j][k]=max(dp[i-1][j][k],dp[i-1][j-1][change(k)]);
}
}
}
}
int M=0;
for(int i=0;i<=T;i++){//最后在整个dp里找到最大值
for(int j=0;j<=W+1;j++){
for(int k=1;k<=2;k++){
M=max(M,dp[i][j][k]);
}
}
}
cout<<M<<endl;
return 0;
}