p1594(巨坑题!!!)

 

护卫车队在一条单行的街道前排成一队,前面河上是一座单行的桥。因为街道是一条单行道,所以任何车辆都不能超车。桥能承受一个给定的最大承载量。为了控制桥上的交通,桥两边各站一个指挥员。护卫车队被分成几个组,每组中的车辆都能同时通过该桥。当一组车队达到了桥的另一端,该端的指挥员就用电话通知另一端的指挥员,这样下一组车队才能开始通过该桥。每辆车的重量是已知的。任何一组车队的重量之和不能超过桥的最大承重量。被分在同一组的每一辆车都以其最快的速度通过该桥。一组车队通过该桥的时间是用该车队中速度最慢的车通过该桥所需的时间来表示的。问题要求计算出全部护卫车队通过该桥所需的最短时间值。

输入格式

输入文件第一行包含三个正整数(用空格隔开),第一个整数表示该桥所能承受的最大载重量(用吨表示);第二个整数表示该桥长度(用千米表示);第三个整数表示该护卫队中车辆的总数(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

 

我恨这题!!!

我恨这题!!!

我恨这题!!!

其实呢,它就是一个普普通通的dp题,本打算练练dp,但没想到居然被坑了一个晚上orz。。。。

题面其实很简单:分段,每段不能超过一个值,每段的的代价是每一段的最大值,求最终总代价的最小值

很好分析,直接枚举断点,方程式秒出来是

dp[i]=min(dp[i],dp[j]+max(a[i]~a[j]));然后前缀和出来就是答案。

但是,我为什么会恨这题呢?

因为它根本不是noip题,它根本就是小学奥数题!!!

1、区间最值,我肯定想到st表啊,秒出来一个中规中矩的st表,然后发现返回值要double...

2、这个破题,这个数据范围,谁会想到它居然要开long long???!!!

3、还要小时转分钟???这不是小学奥数是什么???

啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊!!!!!!!

被这题折磨了整整一晚上。。。

#include<bits/stdc++.h>
using namespace std;
const int maxn=2005;
long long int m,l,n;
struct node
{
    int w;
    double t;
}a[maxn];
int lg[maxn];
double st[maxn][30];
void sst()
{
    lg[0]=-1;
    for(int i=1;i<=n;++i)
    {
        st[i][0]=a[i].t;
        lg[i]=lg[i>>1]+1;
    }
    for(int j=1;j<=25;++j)
    {
        for(int i=1;i+(1<<j)-1<=n;++i)
        {
            st[i][j]=max(st[i][j-1],st[i+(1<<(j-1))][j-1]);
        }
    }
}
double ma(int x,int y)//坑1:返回值是double 
{
    int s=lg[y-x+1];
    return max(st[x][s],st[y-(1<<s)+1][s]);
}
long long int sum[maxn];//坑2:不开ll见祖宗 
double dp[maxn];
int main()
{
    scanf("%lld%lld%lld",&m,&l,&n);
    for(int i=1;i<=n;i++)
    {
        double v;
        scanf("%d%lf",&a[i].w,&v);
        a[i].t=((double)l/v);
        //printf("%lf\n",a[i].t);
        sum[i]=sum[i-1]+a[i].w;
    }
    sst();
    for(int i=1;i<=n;i++)
    dp[i]=a[i].t;
    for(int i=1;i<=n;i++)
    {
        dp[i]+=dp[i-1];
        for(int j=i-1;j>=1;j--)
        {
            if(sum[i]-sum[j-1]<=m)
            {
                dp[i]=min(dp[i],dp[j-1]+ma(j,i));
            }
            else break;
        }
    }
    printf("%.1lf",dp[n]*60);//坑3:小学奥数 
    return 0;
}
 

(完)

posted @ 2019-07-30 22:41  阿基米德的澡盆  阅读(110)  评论(0编辑  收藏  举报