【NOIP2011 Day 2】观光公交

【问题描述】

  小城Y市,拥有n个景点。由于慕名而来的游客越来越多,Y市特意安排了一辆观光公交车,为游客提供更便捷的交通服务。观光公交车在第0分钟出现在1号景点,随后依次前往2、3、4……n号景点。从第i号景点开到第i+1号景点需要Di分钟。任意时刻,公交车只能往前开,或在景点处等待。  设共有m个游客,每位游客需要乘车1次从一个景点到达另一个景点,第i 位游客在Ti分钟来到景点Ai,希望乘车前往景点Bi(Ai

  那么ZZ该如何安排使用加速器,才能使所有乘客的旅行时间总和最小?

  对于100%的数据,1≤n≤1,000,1≤m≤10,000,0≤k≤100,000,0≤Di ≤100,0≤Ti≤100,000。

【分析】

  设t[i]表示来到第i个景点的乘客最晚的时间,time[i]表示车到达第i个景点的最小时间。

  因为每个乘客到达的时间已经固定,所以要使总时间最小,就是使Σtime[b[i]]最小,其中b[i]代表每位乘客的目的地。

  先考虑不用加速器的情况。可以直接递推求出答案,time[i] = max(time[i - 1],t[i - 1]) + d[i - 1];

  接下来再来考虑使用加速器减少的时间。对于每个加速器,我们必须使这个加速器获得最大的效益,即使尽可能多的乘客旅行时间减一。如果我们在i到i + 1间使用加速器的话,那么到i + 1站的乘客的旅行时间都会减一,但是如果time[i + 1]小于t[i + 1]的话,车就要在i + 1站等到t[i + 1]所有的乘客上车,在i + 1站以后下车的乘客的时间是一样的,也就是说这个加速器对后面下车的乘客没有影响。我们可以用递推求出每个i最远能影响的车站。

  g[i] = g[i + 1]    (time[i + 1] > t[i + 1])

  g[i] = i + 1 (time[i + 1] <= t[i + 1])

  那么,我们用一个加速器所能减少一个单位时间的乘客就是sum[g[i]] - sum[i],每次找出使这个最大的i即可。然后把ans减掉,维护time,g

  这题说白了就是贪心。加速器的使用各不影响。易知这个贪心是正确的。

  代码还是比较好写的。

【代码】

#include 
#include 
#include 
#include 
using namespace std;
struct node
{
    int start,arrive,target;
}a[20000];
int n,m,K,ans;
int f[20000],Time[20000],g[20000],dist[20000],sum[20000];
int main()
{
    scanf("%d %d %d",&n,&m,&K);
    for (int i = 1;i < n;i ++)
        scanf("%d",&dist[i]);
    for (int i = 1;i <= m;i ++)
    {
        scanf("%d %d %d",&a[i].arrive,&a[i].start,&a[i].target);
        f[a[i].start] = max(f[a[i].start],a[i].arrive);
        sum[a[i].target] ++ ;
    }
    for (int i = 2;i <= n;i ++)
        sum[i] += sum[i - 1];

    Time[1] = 0;
    for (int i = 2;i <= n;i ++)
        Time[i] = max(Time[i - 1],f[i - 1]) + dist[i - 1];
    for (int i = 1;i <= m;i ++)
        ans += Time[a[i].target] - a[i].arrive;


    while (K)
    {
        g[n] = n;
        g[n - 1] = n;
        for (int i = n - 2;i ; i -- )
        {
            if (Time[i + 1] <= f[i + 1])
                g[i] = i + 1;
            else g[i] = g[i + 1];
        }
        int Max = 0,j;
        for (int i = 1;i <= n;i ++)
            if (sum[g[i]] - sum[i] > Max && dist[i] > 0)
                Max = sum[g[i]] - sum[i],j = i;
        if (!Max) break;
        ans -= Max;
        dist[j] --;
        K --;
        Time[1] = 0;
        for (int i = 2;i <= n;i ++)
            Time[i] = max(Time[i - 1],f[i - 1]) + dist[i - 1];
    }
    cout << ans;
}

 

 

转载于:https://www.cnblogs.com/N-C-Derek/p/3371058.html

你可能感兴趣的:(【NOIP2011 Day 2】观光公交)