11-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
*/
/*思路: 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 <cstring>
#include <cstdio>
using namespace std;
#define MAX   0x3f3f3f3f
int dp[10002][6][6];   //到第几次从i到j的总消耗
int it[6][6];  //从i到j的代价
int a[10002];  //路径

int main(){
    int t;
    it[0][1] = it[0][2] = it[0][3] = it[0][4] = 2; //提前找好各种走法的消耗
    it[1][1] = it[2][2] = it[3][3] = it[4][4] = 1;
    it[1][2] = it[2][1] = it[2][3] = it[3][2] = it[3][4] = it[4][3] = it[4][1] = it[1][4]= 3;
    it[1][3] = it[3][1] = it[2][4] = it[4][2] = 4;
    while(~scanf("%d", &t) && t){
        int n, ans = 0;
        a[1] = t;
        for(n = 2; ;n++){
            scanf("%d", &a[n]);
            if(a[n] == 0){
                n--;
                break;
            }
        }
        memset(dp, MAX, sizeof(dp));
        dp[0][0][0] = 0;
        ans = MAX;
        for(int i = 1; i <= n; i++){
            for(int j = 0; j < 5; j++){ //找一个不动点
                if(a[i] == j)  //这个点,移动最少消耗最少,肯定就是它,故不能是不动点
                    continue;
                int x = MAX, y = MAX;   
                for(int k = 0; k < 5; k++){   //四个踏板向目标点踩   不必担心是不是从前一轮的点,因为dp[][][]初始化为最大值了,非对应的处理时会无穷大,后面比较时候会被淘汰
                    if(k != j || j + k == 0){ //左右脚不能同在非零点
                        x = min(x, dp[i-1][k][j] + it[k][a[i]]);  //左脚踩
                        y = min(y, dp[i-1][j][k] + it[k][a[i]]);  //要脚踩                     
                    }
                }
                dp[i][j][a[i]] = dp[i][a[i]][j] = min(x, y);
                ans = min(ans, dp[n][j][a[i]]);   //找到对应最后一次移动的最小消耗
            }
        }
        printf("%d\n", ans);
    }
    return 0;
}

时间: 2024-10-14 23:26:57

11-st跳舞消耗体力最少的相关文章

青蛙走迷宫问题(体力值)

题目: 青蛙走迷宫,1代表路通,0代表不通:起点是(0, 0),终点是(0,m - 1);青蛙每次向上走需要消耗体力值为3,向下走不消耗体力值,平走消耗体力值1:根据给定值判断青蛙是否可以根据初始值到达终点,并求出消耗体力值最少的路径: 举例: n = 4, m =4, p = 10(体力值) 4 4 10 1 0 0 1 1 1 0 1 0 1 1 1 0 0 1 1 则结果为:[[0, 0], [1, 0], [1, 1], [2, 1], [2, 2], [2, 3], [1, 3], [

3049 舞蹈家怀特先生

3049 舞蹈家怀特先生 时间限制: 1 s 空间限制: 64000 KB 题目等级 : 黄金 Gold 题目描述 Description 怀特先生是一个大胖子.他很喜欢玩跳舞机(Dance Dance Revolution, DDR),甚至希望有一天人家会脚踏“舞蹈家怀特先生”.可惜现在他的动作根本不能称作是在跳舞,尽管每次他都十分投入的表演.这也难怪,有他这样的体型,玩跳舞机是相当费劲的.因此,他希望写一个程序来安排舞步,让他跳起来轻松一些,至少不要每次都汗流浃背. DDR的主要内容是用脚来

常见算法和例题

第3章  算法与程序设计模块 3.1  算    法 算法是对特定问题求解步骤的一种描述,它是指令的有限序列,其中每一条指令表示一个或多个操作. 常用的算法:列举了穷举搜索.递归.回溯.递推.模拟.分治.贪心.深度优先搜索.广度优先搜索等几种较为常用的算法,没有做过多的描述,一旦给出具体描述,容易使内容加深,产生严重学科取向的引导,符合教育部普通高中课程方案的特点,对于这些必需的方法和思想,关键不在于学生能不能,而在于教师是否想到,是否有过关注,引发学生对系统方法和思想的思考,重视建立编程思想,

【算法】搞定[机试]算法刷题 全文超过80页pdf

目录 算法专题 一.树和图 1. 二叉树构造和遍历 2. 朋友圈 - 并查集 3. 公共朋友 - 非朋友圈 4. 哈夫曼树 5. 其他二叉树性质相关计算 6. 图的连通分量 7. 最小生成树 8. 单源最短路径 - dijkstra 二.枚举搜索 1. 按钮开关问题 2. 多层枚举问题 三.递归搜索 1. 简单递归 2. 递增排列组合类 3. 全排列问题 4. 草丛问题 5. 迷宫问题 6. 广度优先搜索的剪枝 7. 总操作步数固定枚举问题 四.数学问题 1. 高精度计算 2. 素数和质因子分解

Laoj P1170 [noip2004]合并果子(STL解法)

试题描述 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆.每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和.可以看出,所有的果子经过n-1次合并之后,就只剩下一堆了.多多在合并果子时总共消耗的体力等于每次合并所消耗体力之和.因为还要花大力气把这些果子搬回家,所以多多在合并果子时要尽可能地节省体力.假定每个果子重量都为1,并且已知果子的种类数和每种果子的数目,你的任务是设计出合并的次序方案,使多多耗费的体力最少

洛谷 1090合并果子

题目描述 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆. 每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和.可以看出,所有的果子经过n-1次合并之后,就只剩下一堆了.多多在合并果子时总共消耗的体力等于每次合并所耗体力之和. 因为还要花大力气把这些果子搬回家,所以多多在合并果子时要尽可能地节省体力.假定每个果子重量都为1,并且已知果子的种类数和每种果子的数目,你的任务是设计出合并的次序方案,使多多耗费的体力最

舞蹈家怀特先生

codevs 3049 舞蹈家怀特先生 http://codevs.cn/problem/3049/ 难度等级:黄金 题目描述 Description 怀特先生是一个大胖子.他很喜欢玩跳舞机(Dance Dance Revolution, DDR),甚至希望有一天人家会脚踏"舞蹈家怀特先生".可惜现在他的动作根本不能称作是在跳舞,尽管每次他都十分投入的表演.这也难怪,有他这样的体型,玩跳舞机是相当费劲的.因此,他希望写一个程序来安排舞步,让他跳起来轻松一些,至少不要每次都汗流浃背. D

[NOIP2004] 提高组 洛谷P1090 合并果子

题目描述 在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆. 每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和.可以看出,所有的果子经过n-1次合并之后,就只剩下一堆了.多多在合并果子时总共消耗的体力等于每次合并所耗体力之和. 因为还要花大力气把这些果子搬回家,所以多多在合并果子时要尽可能地节省体力.假定每个果子重量都为1,并且已知果子的种类数和每种果子的数目,你的任务是设计出合并的次序方案,使多多耗费的体力最

滴滴出行2017秋招工程岗笔试题-地下迷宫

时间限制:1秒 空间限制:32768k 题目描述小青蛙有一天不小心落入了一个地下迷宫,小青蛙希望用自己仅剩的体力值P跳出这个地下迷宫.为了让问题简单,假设这是一个n*m的迷宫.迷宫每个位置为0或者1. 0代表这个位置有障碍物,小青蛙到达不了这个位置:1代表小青蛙可以达到的位置.小青蛙初始在(0,0)位置.地下迷宫的出口在(0,m-1)(保证这两个位置都是1,并且保证一定有起点到终点可达的路径).小青蛙在迷宫中水平移动一个单位距离需要消耗1点体力值.向上爬一个单位距离需要消耗3个体力值,向下移动不