dp[i][j][k]:表示第i次踩踏后两脚的位置j,k
先固定一只脚的位置j,第i次踩踏后,状态为dp[i][j][a[i]]或者dp[i][a[i]][j],其中a[i]表示第i个输入的元素,则有状态方程:
x=dp[i-1][k][j]+cost[k][a[i]]; 是通过k踩过来的,cost[k][a[i]]表示k->a[i]的花费。
y=dp[i-1][j][k]+cost[k][a[i]]; 是通过k踩过来的,cost[k][a[i]]表示k->a[i]的花费。
dp[i][j][a[i]]=dp[i][a[i]][j]=min(x,y);
答案:ans=min(dp[n][j][a[i]])
#include <iostream>
#include <math.h>
using namespace std;
int a[10005],dp[10005][6][6];
int cost[6][6];
int main(int argc, char *argv[])
{
int i,j,k,ans,t,n,x,y;
while(cin>>t&&t)
{
a[1]=t;
for(n=2;;n++)
{
cin>>a[n];
if(!a[n]) break;
}
for(i=0;i<5;i++) for(j=0;j<5;j++) cost[i][j]=100000005;
cost[0][1]=cost[0][2]=cost[0][3]=cost[0][4]=2;
cost[1][2]=cost[2][1]=cost[2][3]=cost[3][2]=cost[3][4]=cost[4][3]=cost[4][1]=cost[1][4]=3;
cost[1][3]=cost[3][1]=cost[2][4]=cost[4][2]=4;
cost[1][1]=cost[2][2]=cost[3][3]=cost[4][4]=1;
for(i=0;i<=n;i++) for(j=0;j<5;j++) for(k=0;k<5;k++) dp[i][j][k]=100000005;
dp[0][0][0]=0;ans=100000005;
for(i=1;i<n;i++)
{
for(j=0;j<5;j++)//没动的脚
{
if(j==a[i]) continue;
x=y=100000005;
for(k=0;k<5;k++) //左脚踩
{
if(k!=j||k+j==0) if(x>dp[i-1][k][j]+cost[k][a[i]]) x=dp[i-1][k][j]+cost[k][a[i]];
}
for(k=0;k<5;k++) //右脚踩
{
if(k!=j||k+j==0) if(y>dp[i-1][j][k]+cost[k][a[i]]) y=dp[i-1][j][k]+cost[k][a[i]];
}
if(x>y) x=y;
dp[i][j][a[i]]=dp[i][a[i]][j]=x;
//cout<<i<<" -> "<<a[i]<<" -> "<<j<<" "<<dp[i][a[i]][j]<<endl;
if(ans>dp[n-1][j][a[i]]) ans=dp[n-1][j][a[i]];
}
}
cout<<ans<<endl;
}
return 0;
}