题目描述:
过河 (20 分)
有N个人想要过一条河,但是他们只有一条最多载两人的船。因此必须想出一个调度船来回的方法让每个人都能过河。每个人都有自己的划船速度,且同一条船上的两个人取决于慢者的速度。你的任务就是想出一个每人都能过河的最快策略。
输入格式:
输入的第一行是一个正整数T(1 <= T <= 20),表示测试用例的组数。下面是T组用例。每个用例的第一行是正整数N,第二行是N个正整数表示每个人的划船速度。每组用例不会超出1000个人,每个人的划船时间不会超过100秒。
输出格式:
对于每个用例,输出所有N个人都能过河的最短时间(秒)。
输入样例:
2
3
1 3 7
4
1 2 5 10
输出样例:
11
17
分析:
我们考虑把单独过河所需要时间最多的两个旅行者送到对岸去,有两种方式:
1.最快的和次快的过河,然后最快的将船划回来;次慢的和最慢的过河,然后次快的将船划回来,所需时间为:
t[0]+2*t[1]+t[n-1];
2.最快的和最慢的过河,然后最快的将船划回来,最快的和次慢的过河,然后最快的将船划回来,所需时间为:
2*t[0]+t[n-2]+t[n-1]。
(这里我们考虑这两种方式的原因,主要是船能乘2人,应该尽量想办法把最慢的两人先送过去,保证最后一定是最快就和次快的人到对岸。这就有两种,一是只有最快的人回来,次快的在岸上等;二是最快和最慢的人都回来一次,也就是第一次先把次快的送过去,倒数第二次,次快的再回来和最快的一起过去。)
当n大于4是,重复该情况即可。
代码:
#include<iostream>
using namespace std;
#include<cstring>
#include<string>
#include<cctype>
#include<cmath>
#include<vector>
#include<algorithm>
int a[1005];
int crossRiver(int n,int*a)
{
if(n==0) return 0;
if(n==1) return a[0];
else if(n==2) return a[1];
else if(n==3) return a[0]+a[1]+a[2];
else
{
int t1=a[n-1]+a[0]+a[n-2]+a[0];
int t2=a[1]+a[0]+a[n-1]+a[1];
int t;
t=t1>t2?t2:t1;
return t+crossRiver(n-2,a);
}
}
int main()
{
int t;
cin>>t;
while(t--)
{
int n;
cin>>n;
for(int i=0;i<n;i++)
cin>>a[i];
sort(a,a+n);
int sum=crossRiver(n,a);
cout<<sum<<endl;
memset(a,0,sizeof(a));
}
return 0;
}
版权声明:本文为qq_40725780原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。