题目 740 “炫舞家“ST

“炫舞家“ST

时间限制: 3000 ms  |  内存限制: 65535 KB
难度: 3
描述
ST是一个酷爱炫舞的玩家。TA很喜欢玩QQ炫舞,因此TA也爱屋及乌的喜欢玩跳舞机(Dance Dance Revolution,DDR)。但是TA每天还要努力的学习,因此TA希望每次都保存最多的体力来学习。

DDR的主要内容是用脚来踩踏板。踏板有4个方向的箭头,用1,2,3,4来代表,如下图所示。

                                                    

 游戏规则如下:
   每首歌曲有一个箭头序列,游戏者必须按照这个序列依次用某一只脚踩相应的踏板。在任何时候,两只脚都不能在同一个踏板上,但可以同时待在中心位置0(一开始游戏的时候,游戏者的双脚都在中心位置0处)。
   每一个时刻,TA必须移动而且只能移动TA的一只脚去踩相应的箭头,而另一只脚不许移动。这样,TA跳DDR的方式可以用一串数字L1L2………Ln来表示。
   其中体力消耗规则如下:
1、 从中心往任何一个箭头耗费2个单位体力;
2、 从任何一个箭头移动到相邻箭头耗费3个单位体力(1和3相对,2和4相对)耗费4个单位体力。
3、 留在原地在踩一下只需要1单位。 
现在炫舞家ST很想学习但是又想玩DDR。因此,TA希望厉害的程序员你可以帮TA编写一个程序计算出TA因该怎样移动他的双脚(即,对于每个箭头,选一只脚去踩它),才能用最少的体力完成给定的舞曲。
例如,给出22140,总的体力耗费为2+1+2+3=8单位。 

输入
输入文件将包括一系列的方向序列。每个方向序列包含一个数字序列。每个输入序列应该是数字1、2、3或4,每个代表四个方向之一。一个值为0的方向序列表示方向的结束序列。和这个值应该被排除在方向序列(每个方向序列输入最多包含10000个数字)。输入文件结束为输入序列只有单独的一个0。
输出
对于每个方向序列,输出最少单位的体力消耗值。结果应该是一个整数在单独的一行。任何多余的白空格或空行将不被接受。
样例输入
2 3 3 3 3 1 2 0
3 2 2 1 2 0
0
样例输出
12
9
来源
BE玩家
上传者
ACM_钟诗俊


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)

#include <iostream>
#include <math.h>
using namespace std;
int a[10005],dp[10005][6][6];
int cost[6][6];
int main()
{
	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]==0) 
				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;
}
初始化可以借助STL;
#include<iostream>
#include<string.h>
using namespace std;
int dp[10005][6][6];
int cost[6][6];
int a[10005];

int main()
{
	int t;
	while(cin>>t&&t)
	{
		a[1]=t;
		int n;
		for( n=2;;n++)
		{
			cin>>a[n];
			if(a[n]==0)
				break;
		}
		memset(dp,10,sizeof(dp));
		
		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;  
		dp[0][0][0]=0;
		int x,y,k,min=dp[1][1][1];
		
		for(int i=1;i<n;i++)
			{
				for(int j=0;j<5;j++)
				{
					if(j==a[i])
						continue;
					x=y=min;
					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;
					if(min>dp[n-1][j][a[i]])
						min=dp[n-1][j][a[i]];

				}
			}
		cout<<min<<endl;
	}
	return 0;
}



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值