护卫队(动态规划+ST表)

护卫队

题目描述:
护卫车队在一条单行的街道前排成一队,前面河上是一座单行的桥。因为街道是一条单行道,所以任何车辆都不能超车。桥能承受一个给定的最大承载量。为了控制桥上的交通,桥两边各站一个指挥员。护卫车队被分成几个组,每组中的车辆都能同时通过该桥。当一组车队达到了桥的另一端,该端的指挥员就用电话通知另一端的指挥员,这样下一组车队才能开始通过该桥。每辆车的重量是已知的。任何一组车队的重量之和不能超过桥的最大承重量。被分在同一组的每一辆车都以其最快的速度通过该桥。一组车队通过该桥的时间是用该车队中速度最慢的车通过该桥所需的时间来表示的。问题要求计算出全部护卫车队通过该桥所需的最短时间值。
输入输出格式:
输入格式:
输入文件第一行包含三个正整数(用空格隔开),第一个整数表示该桥所能承受的最大载重量(用吨表示);第二个整数表示该桥长度(用千米表示);第三个整数表示该护卫队中车辆的总数(n<1000)。接下来的几行中,每行包含两个正整数W和S(用空格隔开),W表示该车的重量(用吨表示),S表示该车过桥能达到的最快速度(用千米/小时表示)。车子的重量和速度是按车子排队等候时的顺序给出的。
输出格式:
输出文件应该是一个实数,四舍五入精确到小数点后1位,表示整个护卫车队通过该桥所需的最短时间(用分钟表示)。
输入输出样例:
输入样例#1:
100 5 10
40 25
50 20
50 20
70 10
12 50
9 70
49 30
38 25
27 50
19 70
输出样例#1:
75.0
思路:
注意开long long
很容易想出暴力搜索的策略:
设当前点为x,枚举从x出发,能到达的点(这些点肯定是连续的,每辆车的重量用前缀和优化),y为到当前点x所用的时间,当x==n时,ans=min(ans,y)
这样就可以拿到40分了

#include<iostream>
#include<cstdio>
#define lon long long
using namespace std;
const int maxn=1010;
lon w,t,n,sum[maxn];
double ans=0x7fffffff;
struct node
{
    lon w;
    lon s;
    double t;
}a[maxn];
void dfs(lon x,double y)
{
    if(x==n)
    {
        ans=min(ans,y);
        return;
    }
    if(y>ans) return;
    for(lon i=x+1;i<=n;i++)
    if(sum[i]-sum[x]<=w)
    {
        double tmp=-0x7fffffff;
        for(lon j=x+1;j<=i;j++)
        tmp=max(a[j].t,tmp);
        dfs(i,y+tmp);
    }
    else break;
}
int main()
{
    scanf("%d%d%d",&w,&t,&n);
    for(lon i=1;i<=n;i++)
    {
        scanf("%d%d",&a[i].w,&a[i].s);
        a[i].t=double(t)/double(a[i].s);
        sum[i]=sum[i-1]+a[i].w;
    }
    dfs(0,0);
    printf("%0.1lf",ans*60);
    return 0;
}

根据这个思路,转成动态规划就好了,ST表O(1)查询两点之间所需要的时间。
这样就可以AC了

#include<iostream>
#include<cstdio>
#include<cmath>
#define lon long long
using namespace std;
const int maxn=1010;
lon w,t,n,sum[maxn];
double ans=0x7fffffff,f[maxn],b[maxn][20];
struct node
{
    lon w;
    lon s;
    double t;
}a[maxn];
void prepare()
{
    for(int j=1;j<=19;j++)
      for(int i=1;(i+(1<<j)-1<=n);i++)
      b[i][j]=max(b[i][j-1],b[i+(1<<j-1)][j-1]);
}
double maxx(lon l,lon r)
{
    double t;
    lon k=double(log(r-l+1.0))/double(log(2.0));
    t=max(b[l][k],b[r-(1<<k)+1][k]);
    return t;
}
int main()
{
    scanf("%d%d%d",&w,&t,&n);
    for(lon i=1;i<=n;i++)
    {
        scanf("%d%d",&a[i].w,&a[i].s);
        f[i]=double(t)/double(a[i].s);
        sum[i]=sum[i-1]+a[i].w;
        b[i][0]=f[i];
    }
    prepare();
    for(int i=1;i<=n;i++)//前i辆车最小时间 
    {
        f[i]+=f[i-1];
        for(int j=i-1;j>=1;j--)//由第j辆车到第i辆车 
        {
            if(sum[i]-sum[j-1]>w) break;
            f[i]=min(f[i],f[j-1]+maxx(j,i));
        }
    }
    printf("%0.1lf",f[n]*60);
    return 0;
}
posted @ 2016-11-16 08:55  抽空的太阳  阅读(370)  评论(0编辑  收藏  举报