• WILL吃桃_KEY


    WILL 吃桃

    (peach.pas/c/cpp)

    【 题目描述】

    Will 很喜欢吃桃, 某天 Will 来到了一片森林, 森林中有 N 颗桃树, 依次编号为 1,2,„,N.每棵树上有数量不等的桃子。 某些桃树之间有单向通行的小路, 且路径不会形成环, 通过每条小路的时间也不一定相同。 现在, Will 提着一个最多可以容纳 K 个桃子的篮子, 从编号为1 的桃树出发, 走过若干条小路之后来到编号为 N 的桃树。 当 Will 在路上走的时候, 每走 1分钟, 他会从篮子中拿出一个桃子来吃掉( 如果篮子中还有桃子的话, 如果篮子中没有桃子的话那就没得吃了!)。 每到一棵桃树( 包括起点和终点), 他会把这棵桃树上的所有桃子摘下来放入篮子中。 现在你的问题是: 求 K 的最小值, 使得 Will 能够不浪费任何桃子( 每到一棵桃树, 这棵树上的所有桃子都必须被装入篮子中)。

    【 输入格式】

    输入文件第一行两个整数, N,m, 分别表示桃树的数量以及连接桃树的小路的数量。

    接下来一行 N 个用空格隔开的整数, 分别表示每一颗桃树上的桃子的数量。

    接下来 m 行, 每行 3 个用空格隔开的整数, a,b,c, 表示有一条小路能够从桃树 a 走到桃树 b,( 注意小路一定是单向的), 走过这条小路所需要的时间是 c 分钟。从任意一棵桃树出发, Will 不可能沿着小路走若干条路之后重新回到这棵桃树。( 给出的图是一个有向无环图。) 数据保证 Will 一定能够从桃树 1 走到桃树 N。

    【 输出格式】

    输出文件有且仅有一行, 一个整数, 表示 K 的最小值

    【 输入样例】

    3 3

    5 1 6

    1 3 1

    1 2 4

    2 3 5

    【 输出样例】

    6

    【 数据规模】

    对于 30%的数据: 3≤N≤10; m≤20;

    对于 60%的数据: 3≤N≤1,000; m≤10,000;

    对于 100%的数据: 3≤N≤10,000; 3≤m≤30,000; 所有其他数据都不超过 10000;

    乍一看,数据很大,所以在这儿我们可以二分答案,SPFA判断是否可行。

    code

    #include <cstdio>
    #include <vector>
    #include <cstring>
    #define mm 120000
    using namespace std;
    int n,m,l[120001],h,t,vis[10001],p[10001],ans=2e9,f[10001];
    vector <pair<int,int> >a[10001];
    inline int min(int x,int y){return x<y?x:y;}
    inline int max(int x,int y){return x>y?x:y;}
    int SPFA(int wks){
        memset(vis,0,sizeof(vis));
        memset(f,63,sizeof(f));
        h=t=0;
        l[++t]=1,f[1]=p[1];
        vis[1]=1;
            while(h<t){
                h+=1;
                int front=l[h];vis[front]=0;
                    for(int i=0;i<a[front].size();i++){
                        int to=a[front][i].first;
                        if(p[to]+max(0,f[front]-a[front][i].second)<f[to]&&p[to]+max(0,f[front]-a[front][i].second)<=wks){
                            f[to]=p[to]+max(0,f[front]-a[front][i].second);
                            if(!vis[to])vis[to]=1,l[t=t+1]=to;
                        }
                    }
            }
        return f[n]<=wks;
    }
    int main(){
        freopen("peach.in","r",stdin);
        freopen("peach.out","w",stdout);
        scanf("%d%d",&n,&m);
            for(int i=1;i<=n;i++)scanf("%d",&p[i]);
            for(int i=1;i<=m;i++){
                int x,y,c;scanf("%d%d%d",&x,&y,&c);
                a[x].push_back(make_pair(y,c));
            }
        int l=p[1],r=100000005;
            while(l<r){
                int mid=(l+r)>>1;
                if(!SPFA(mid))l=mid+1;
                else r=mid;
            }
        printf("%d",l);
        fclose(stdin),fclose(stdout);
        return 0;
    }
  • 相关阅读:
    记忆力训练今天早上有了点小进步
    刻意练习
    12.12周计划
    12.6周总结
    The Power of Reading Insights
    Storytelling with Data
    nexus私服和快照正式版本etc
    springboot启动流程分析
    容器启动getBean的流程分析
    canal简介
  • 原文地址:https://www.cnblogs.com/Cptraser/p/7593454.html
Copyright © 2020-2023  润新知