接水问题(贪心课后题顺序查找是什么鬼)

15:接水问题

总时间限制: 
1000ms
 
内存限制: 
65536kB
描述

学校里有一个水房,水房里一共装有 m 个龙头可供同学们打开水,每个龙头每秒钟的供水量相等,均为 1。

现在有 n 名同学准备接水,他们的初始接水顺序已经确定。将这些同学按接水顺序从 1 到 n 编号,i号同学的接水量为 wi。接水开始时,1 到 m 号同学各占一个水龙头,并同时打开水龙头接水。当其中某名同学 j 完成其接水量要求 wj后,下一名排队等候接水的同学 k 马上接替 j 同学的位置开始接水。这个换人的过程是瞬间完成的,且没有任何水的浪费。即 j 同学第 x 秒结束时完成接水,则 k 同学第 x+1 秒立刻开始接水。 若当前接水人数 n’不足 m,则只有 n’个龙头供水,其它 m-n’个龙头关闭。

现在给出 n 名同学的接水量,按照上述接水规则,问所有同学都接完水需要多少秒。

输入
第 1 行2 个整数 n 和 m,用一个空格隔开,分别表示接水人数和龙头个数。
第 2 行 n 个整数 w1、w2、……、wn,每两个整数之间用一个空格隔开,wi表示 i 号同学的接水量。

1 ≤ n ≤ 10000,1 ≤ m ≤ 100 且 m ≤ n;
1 ≤ wi ≤ 100。
输出
输出只有一行,1 个整数,表示接水所需的总时间。
样例输入
样例 #1:
5 3
4 4 1 2 1

样例 #2:
8 4
23 71 87 32 70 93 80 76
样例输出
样例 #1:
4

样例 #2:
163
提示
输入输出样例1解释:
第 1 秒,3 人接水。第 1秒结束时,1、2、3 号同学每人的已接水量为 1,3 号同学接完水,4 号同学接替 3 号同学开始接水。
第 2 秒,3 人接水。第 2 秒结束时,1、2 号同学每人的已接水量为 2,4 号同学的已接水量为 1。
第 3 秒,3 人接水。第 3 秒结束时,1、2 号同学每人的已接水量为 3,4 号同学的已接水量为 2。4号同学接完水,5 号同学接替 4 号同学开始接水。
第 4 秒,3 人接水。第 4 秒结束时,1、2 号同学每人的已接水量为 4,5 号同学的已接水量为 1。1、2、5 号同学接完水,即所有人完成接水。
总接水时间为 4 秒。
来源
NOIP2010复赛 普及组 第二题
【神的分析】感觉我思路有点奇葩,可是就是这么推出来的呀,,只是代码实现了而已。
【代码妹子】
 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstdlib>
 4 #include<algorithm>
 5 using namespace std;
 6 int Time[10009];
 7 int n,m,maxx=0;
 8 int main()
 9 {
10     scanf("%d%d",&n,&m);
11     for(int i=1;i<=n;i++)
12     {
13         scanf("%d",&Time[i]);
14         maxx=max(maxx,Time[i]);//接水时间最长的找出来 
15     }
16     if(m>=n)
17     {
18         printf("%d",maxx);//如果水龙头够分的话时间就是最长时间,输出最长时间结束程序就好了 
19         return 0;
20     }
21     int l=1,r=m,tot=0;//如果水龙头不够就挨个轮呗 
22     for(int j=l;j<=n-m+1;j++)//m个水龙头从左向右移动,也就是长为m的区间平移 
23     {
24         sort(Time+l,Time+r+1);//排序后的第一个就是花费最短时间的 
25         if(r==n)//建议先看完下面的for循环再看这个if语句 
26         {
27             tot+=Time[r];//当右移到头时,最后m个人一定是花费时间一定是m个人中的最长时间 
28             printf("%d",tot);//所以tot要加入排序后的最后一个 
29             return 0;//结束 
30         }
31         for(int i=l+1;i<=r;i++)//注意这里一定要从l+1开始循环,因为要减去Time[l],不能让它的值先改变改变 
32         {
33             Time[i]-=Time[l];//每个人的时间减去Time[l],也就是减去这m个水龙头中最少的时间 
34         }
35         tot+=Time[l];//总时间加上这个刚走的人的时间(刚走的人就是排序后最小的呀) 
36         l++;r++;//区间向右移动,继续下一个人; 
37     }
38     return 0;
39 }

 

posted @ 2017-03-30 16:40  ANhour  阅读(884)  评论(0编辑  收藏  举报