您的位置:首页 > 其它

noip2009道路游戏【高诗豪】

2016-11-17 14:25 357 查看
题目:


题目描述

小新正在玩一个简单的电脑游戏。

游戏中有一条环形马路,马路上有 n 个机器人工厂,两个相邻机器人工厂之间由一小段马路连接。小新以某个机器人工厂为起点,按顺时针顺序依次将这 n 个机器人工厂编号为1~n,因为马路是环形的,所以第 n 个机器人工厂和第 1 个机器人工厂是由一段马路连接在一起的。小新将连接机器人工厂的这 n 段马路也编号为 1~n,并规定第 i 段马路连接第 i 个机器人工厂和第 i+1 个机器人工厂(1≤i≤n-1),第
n 段马路连接第 n 个机器人工厂和第 1个机器人工厂。

游戏过程中,每个单位时间内,每段马路上都会出现一些金币,金币的数量会随着时间发生变化,即不同单位时间内同一段马路上出现的金币数量可能是不同的。小新需要机器人的帮助才能收集到马路上的金币。所需的机器人必须在机器人工厂用一些金币来购买,机器人一旦被购买,便会沿着环形马路按顺时针方向一直行走,在每个单位时间内行走一次,即从当前所在的机器人工厂到达相邻的下一个机器人工厂,并将经过的马路上的所有金币收集给小新,例如,小新在 i(1≤i≤n)号机器人工厂购买了一个机器人,这个机器人会从
i 号机器人工厂开始,顺时针在马路上行走,第一次行走会经过 i 号马路,到达 i+1 号机器人工厂(如果 i=n,机器人会到达第 1 个机器人工厂),并将 i 号马路上的所有金币收集给小新。 游戏中,环形马路上不能同时存在 2 个或者 2 个以上的机器人,并且每个机器人最多能够在环形马路上行走 p 次。小新购买机器人的同时,需要给这个机器人设定行走次数,行走次数可以为 1~p 之间的任意整数。当马路上的机器人行走完规定的次数之后会自动消失,小新必须立刻在任意一个机器人工厂中购买一个新的机器人,并给新的机器人设定新的行走次数。

以下是游戏的一些补充说明:

游戏从小新第一次购买机器人开始计时。

购买机器人和设定机器人的行走次数是瞬间完成的,不需要花费时间。

购买机器人和机器人行走是两个独立的过程,机器人行走时不能购买机器人,购买完机器人并且设定机器人行走次数之后机器人才能行走。

在同一个机器人工厂购买机器人的花费是相同的,但是在不同机器人工厂购买机器人的花费不一定相同。

购买机器人花费的金币,在游戏结束时再从小新收集的金币中扣除,所以在游戏过程中小新不用担心因金币不足,无法购买机器人而导致游戏无法进行。也因为如此,游戏结束后,收集的金币数量可能为负。

现在已知每段马路上每个单位时间内出现的金币数量和在每个机器人工厂购买机器人需要的花费,请你告诉小新,经过 m 个单位时间后,扣除购买机器人的花费,小新最多能收集到多少金币。


输入输出格式

输入格式:

第一行 3 个正整数,n,m,p,意义如题目所述。

接下来的 n 行,每行有 m 个正整数,每两个整数之间用一个空格隔开,其中第 i 行描

述了 i 号马路上每个单位时间内出现的金币数量(1≤金币数量≤100),即第 i 行的第 j(1≤j≤m)个数表示第 j 个单位时间内 i 号马路上出现的金币数量。

最后一行,有 n 个整数,每两个整数之间用一个空格隔开,其中第 i 个数表示在 i 号机器人工厂购买机器人需要花费的金币数量(1≤金币数量≤100)。

输出格式:

共一行,包含
1 个整数,表示在 m 个单位时间内,扣除购买机器人

花费的金币之后,小新最多能收集到多少金币。

输入输出样例

输入样例#1:

2 3 2
1 2 3
2 3 4
1 2


输出样例#1:

5











说明

【数据范围】

对于 40%的数据,2≤n≤40,1≤m≤40。

对于 90%的数据,2≤n≤200,1≤m≤200。

对于 100%的数据,2≤n≤1000,1≤m≤1000,1≤p≤m。

做法:题目呢,比较长,信息比较多,注意不要看错题。但是呢还是比较轻易可以看出这是dp题的类型。dp[i][j]表示时间i在j点的最大收益,pre[j]表示j点的上一个,mx[i]表示在时间i所有位置的最大收益(因为买机器人是任意位置可买,转移时直接用即可),g[i][j]表示状态(i,j)取最优解时走的步数(这个明显是越小越好啦),最后输出max(dp[m][i])。

so,代码如下:

[cpp] view
plain copy

#include<iostream>  

#include<algorithm>  

#include<cstdio>  

#include<cstdlib>  

#include<cstring>  

#include<cmath>  

#define maxn 1020  

#define inf 0x3f3f3f  

using namespace std;  

int dp[maxn][maxn];  

int mon[maxn][maxn];  

int g[maxn][maxn];  

int pre[maxn];  

int mx[maxn];  

int c[maxn];  

int n,m,p;  

int main()  

{  

    freoepn("game.in","r",stdin);  

    freoepn("game.out","w",stdout);  

    mx[0]=0;  

    memset(dp,-0x3f,sizeof(dp));//初始化  

    memset(mx,-0x3f,sizeof(mx));  

    scanf("%d%d%d",&n,&m,&p);  

    for(int i=1;i<=n;i++)  

        for(int j=1;j<=m;j++)scanf("%d",&mon[i][j]);  

    for(int i=1;i<=n;i++)  

    {  

        scanf("%d",&c[i]);  

        if(i==1)pre[i]=n;  

        else pre[i]=i-1;  

        dp[0][i]=0;  

    }  

    for(int i=1;i<=m;i++)  

        for(int j=1;j<=n;j++)  

        {  

                if(g[i-1][pre[j]]<p&&i>1){<span style="white-space:pre">  </span>//机器人再走一步的情况转移  

                if(dp[i][j]<dp[i-1][pre[j]]+mon[pre[j]][i])  

                {  

                    dp[i][j]=dp[i-1][pre[j]]+mon[pre[j]][i];  

                    mx[i]=max(mx[i],dp[i][j]);  

                    g[i][j]=g[i-1][pre[j]]+1;  

                }  

                }  

                if(dp[i][j]<mx[i-1]+mon[pre[j]][i]-c[pre[j]])//买一个新的机器人的情况  

                {  

                    dp[i][j]=mx[i-1]+mon[pre[j]][i]-c[pre[j]];  

                    mx[i]=max(mx[i],dp[i][j]);  

                    g[i][j]=1;  

                }  

        }  

    int ans=-inf;  

    for(int i=1;i<=n;++i) ans=max(ans,dp[m][i]);//输出  

    printf("%d",ans);   

    return 0;  

}  
(从转载blog.csdn.net/Fgaoshihao处转载)
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: