BZOJ.1003.[ZJOI2006]物流运输(DP 最短路Dijkstra)

题目链接

容易看出是个最短路+DP。既然答案和天数有关,那么就令f[i]表示前i天最小成本。
这个转移很好想: f[i]=min(f[i],f[j]+cost(j+1,i)+K),cost(j+1,i)即第j+1天到第i天(使用同一道路)所需花费,即最短路,这个可以预处理出来。
注意是否可行的判断。

//880kb 88ms
//好像很吉利啊
#include <queue>
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
#define pr std::pair<int,int>
#define mp std::make_pair
const int N=23,M=900,INF=0x3f3f3f3f;//边数n*n啊。。

int d,n,K,m,Enum,H[N],nxt[M],to[M],len[M],cost[102][102],dis[N],f[102];
bool vis[N],can_t[N][102];
std::priority_queue<pr> q;

inline int read()
{
    int now=0;register char c=gc();
    for(;!isdigit(c);c=gc());
    for(;isdigit(c);now=now*10+c-'0',c=gc());
    return now;
}
inline void AddEdge(int u,int v,int w)
{
    to[++Enum]=v, nxt[Enum]=H[u], H[u]=Enum, len[Enum]=w;
    to[++Enum]=u, nxt[Enum]=H[v], H[v]=Enum, len[Enum]=w;
}
inline bool Judge(int x,int u,int v){
//  return st[x]>v||ed[x]<u;//WA:可能有多段!有毒!
    for(int i=u; i<=v; ++i) if(can_t[x][i]) return 0;
    return 1;
}
int Dijkstra(int a,int b)
{
    memset(vis,0,sizeof vis);
    memset(dis,0x3f,sizeof dis);
    while(!q.empty()) q.pop();
    dis[1]=0, q.push(mp(0,1));
    while(!q.empty())
    {
        int x=q.top().second;q.pop();
        if(x==n) return dis[n]*(b-a+1);
        if(vis[x]) continue;
        vis[x]=1;
        for(int v,i=H[x]; i; i=nxt[i])
            if(!vis[v=to[i]] && Judge(v,a,b) && dis[v]>dis[x]+len[i])
                dis[v]=dis[x]+len[i], q.push(mp(-dis[v],v));
    }
    return INF;
}

int main()
{
    d=read(),n=read(),K=read(),m=read();
    int u,v;
    while(m--) u=read(),v=read(),AddEdge(u,v,read());
    int t=read(),p;
    while(t--){
        p=read(),u=read(),v=read();
        for(int i=u; i<=v; ++i) can_t[p][i]=1;
    }
    for(int i=1; i<=d; ++i)
        for(int j=i; j<=d; ++j)
        {
            cost[i][j]=Dijkstra(i,j);
            if(cost[i][j]==INF){
                for(int k=j+1; k<=d; ++k) cost[i][k]=INF;
                break;
            }
        }
    for(int i=1; i<=d; ++i) f[i]=cost[1][i];
    for(int i=2; i<=d; ++i)
        for(int j=i-1; j; --j)
            if(cost[j+1][i]<INF)//注意INF的判断。
                f[i]=std::min(f[i],f[j]+cost[j+1][i]+K);
            else break;
    printf("%d",f[d]);

    return 0;
}

原文地址:https://www.cnblogs.com/SovietPower/p/8757280.html

时间: 2025-01-12 10:44:08

BZOJ.1003.[ZJOI2006]物流运输(DP 最短路Dijkstra)的相关文章

bzoj 1003: [ZJOI2006]物流运输trans 最短路+dp

题目链接 1003: [ZJOI2006]物流运输trans Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 5246  Solved: 2157[Submit][Status][Discuss] Description 物流公司要把一批货物从码头A运到码头B.由于货物量比较大,需要n天才能运完.货物运输过程中一般要转停好几个码头.物流公司通常会设计一条固定的运输路线,以便对整个运输过程实施严格的管理和跟踪.由于各种因素的存在,有的时候某个码头会无

bzoj 1003 [ZJOI2006]物流运输(最短路+dp)

[ZJOI2006]物流运输 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 8973  Solved: 3839[Submit][Status][Discuss] Description 物流公司要把一批货物从码头A运到码头B.由于货物量比较大,需要n天才能运完.货物运输过程中一般要转停好几个码头.物流公司通常会设计一条固定的运输路线,以便对整个运输过程实施严格的管理和跟踪.由于各种因素的存在,有的时候某个码头会无法装卸货物.这时候就必须修改运输

BZOJ 1003: [ZJOI2006]物流运输trans

二次联通门 : BZOJ 1003: [ZJOI2006]物流运输trans /* BZOJ 1003: [ZJOI2006]物流运输trans Spfa + Dp Spfa预处理出i到j天的最小花费 然后N^2 dp即可 */ #include <cstdio> #include <iostream> #include <cstring> #include <queue> #define INF 1e6 const int BUF = 12312313;

BZOJ 1003: [ZJOI2006]物流运输trans DP+最短路

Description 物流公司要把一批货物从码头A运到码头B.由于货物量比较大,需要n天才能运完.货物运输过程中一般要转停好几个码头.物流公司通常会设计一条固定的运输路线,以便对整个运输过程实施严格的管理和跟踪.由于各种因素的存在,有的时候某个码头会无法装卸货物.这时候就必须修改运输路线,让货物能够按时到达目的地.但是修改路线是一件十分麻烦的事情,会带来额外的成本.因此物流公司希望能够订一个n天的运输计划,使得总成本尽可能地小. Input 第一行是四个整数n(1<=n<=100).m(1&

BZOJ 1003: [ZJOI2006]物流运输(spfa+dp)

http://www.lydsy.com/JudgeOnline/problem.php?id=1003 题意: 思路: 首先用spfa计算一下任意两天之内的最短路,dis[a][b]表示的就是在第a天~第b天从1到m的最短路. 接下来就是dp了,f[i]表示前i天的最小代价,那么状态转移方程就是: f[i]=min(f[i],f[j]+dis[j+1][i]*(i-j)+k) 注意:边界条件f[0]=-k! 1 #include<iostream> 2 #include<algorit

BZOJ 1003: [ZJOI2006]物流运输trans SPFA+DP

原题链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1003 题解: dp就好,令dp[i]表示第i天的答案,那么dp[i]=min{Cost(1,i),Cost(j+1,i)+dp[j]+K},其中Cost(i,j)表示从i到j都用同一种方案.这种dp和划分问题很类似. 代码: #include<iostream> #include<cstring> #include<algorithm> #include<

[bzoj]1003: [ZJOI2006]物流运输

Description 物流公司要把一批货物从码头A运到码头B.由于货物量比较大,需要n天才能运完.货物运输过程中一般要转停好几个码头.物流公司通常会设计一条固定的运输路线,以便对整个运输过程实施严格的管理和跟踪.由于各种因素的存在,有的时候某个码头会无法装卸货物.这时候就必须修改运输路线,让货物能够按时到达目的地.但是修改路线是一件十分麻烦的事情,会带来额外的成本.因此物流公司希望能够订一个n天的运输计划,使得总成本尽可能地小. Input 第一行是四个整数n(1<=n<=100).m(1&

BZOJ 1003 ZJOI2006 物流运输trans 动态规划+SPFA

题目大意:给定一个无向图,运输n天,其中有些天有些点不能走,更换路线代价为k,求代价总和 首先令cost[i][j]为第i天到第j天都走同一路线的最小花销 这个用SPFA处理 然后就是动规的问题了 令f[i]为1~i天的最小花销 则f[i]=min{ f[j]+cost[j+1][i]+k } ( 0<=j<i ) 注意m和n别写反 乘天数之前要特判是不是正无穷 #include<cstdio> #include<cstring> #include<iostrea

1003: [ZJOI2006]物流运输 最短路+dp

https://www.lydsy.com/JudgeOnline/problem.php?id=1003 数据范围很小,怎么瞎搞都行,n方dp,然后跑出最短路暴力转移,需要注意的是不能使用的可能有多个区间 /************************************************************** Problem: 1003 User: walfy Language: C++ Result: Accepted Time:180 ms Memory:1400 k