动态规划——租用游艇问题

问题:

  长江游艇俱乐部在长江上设置了n个游艇出租站1,2,3…,n。

  游客可以在这些游艇出租站用游艇,并在下游的任何一个游艇出租站归还游艇。

  游艇出租站i到游艇出租站j之间的租金为r(i,j),1<=i<j=n。

  试设计一个算法,计算从游艇出租站1到出租站n所需的最少租金。

数据输入:

  第一行表示有n个站点。 

  接下来n-1行是r( i ,  j)。

  输入示例(有3个站点,从 1 到 2 要 5,从 1 到 3 要 15,从 2 到 3 要 7):

    3

    5  15

    7

数据输出:

  输出最从游艇出租站1到出租站n所需的最少租金。

分析:

  假设从第 i 站到第 j 站的最优办法是:从 i 到 k (i<=k<=j),再从 k 到 j ,即 r( i , k )+r( k , j )。则用r( i , k )+r( k , j )代替r( i , j )。

算法思路:

  初始化,先定义一个二维数组m[][],将输入的数据r(i,j)存至m[i][j]。

  接着从长度为 2 的开始找较优解(比如说从 1 到 2 就是长度为 2的),直到找到长度为 n 的。具体做法就是分析中所说的。

  这样计算后,所有数据都在 m 数组中了,我们只要输出 m[1][n]即可。

  计算顺序:

 

ps:我发现动态规划的题目都很类似,只要掌握了其特点(我觉得就是用一个数组来表示解,然后在循环中依据子问题不断计算上一层的父问题),就很好解决。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=205;
int m[maxn][maxn];
int main()
{
    ios::sync_with_stdio(false);
//    freopen("in.txt","r",stdin);
    int n;
    cin>>n;
    memset(m,0,sizeof(m));
    for(int i=1;i<=n-1;i++)//初始化,将所有r(i,j)都先存在数组m中
        for(int j=i+1;j<=n;j++)
            cin>>m[i][j];
    for(int r=2;r<=n;r++)//对角线循环
    {
        for(int i=1;i<=n-r+1;i++)//行循环
        {
            int j=i+r-1;
            for(int k=i;k<=j;k++)//在 i 到 j 中找某一站 k,使得r(i,k)+r(k,j)最小
            {
                int t=m[i][k]+m[k][j];
                if(t<m[i][j])
                {
                    m[i][j]=t;
                }
            }
        }
    }
    cout<<m[1][n]<<endl;
    return 0;
}

转载:https://www.cnblogs.com/jacklovelol/articles/6013111.html

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值