BZOJ 1395 [Baltic2005]Trip(最短路+DP)

【题目链接】 http://www.lydsy.com/JudgeOnline/problem.php?id=1354

【题目大意】

  给出一些车的班次,包括起点,终点,到达起点时间区间,
  到达终点时间区间,想要T时刻到达n号点,问最坏情况下的最短等待时间

【题解】

  最坏情况就是每次从b时刻才出发,c时刻到达,
  那么就相当于地点从x到y,时间从a到d,代价为c-b,
  我们求出符合要求的最大代价,然后用最终时间T去减即可
  如果没有a和d这个限制,可以直接求最长路即可,
  现在考虑如何消除a和d这个时间限制,我们将每条路拆分为两个点
  a点处理答案保存,d点处理最长路的计算,
  按照时间节点排序,顺序求最长路即可。  

【代码】

#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
const int N=50010;
int n,m,T,t,x,y,a,b,c,d,tot,dis[N],ans[N<<1];
struct ask{int x,t,id,dis;}q[N<<2];
bool cmp(ask a,ask b){return a.t==b.t?a.dis>b.dis:a.t<b.t;}
int main(){
    scanf("%d%d%d%d",&n,&m,&T,&t);
    for(int i=1;i<=m;i++){
        scanf("%d%d%d%d%d%d",&x,&y,&a,&b,&c,&d);
        q[++tot]=(ask){x,a,i,0};
        q[++tot]=(ask){y,d,i,c-b};
    }q[++tot]=(ask){n+1,t,0,-2e9};
    sort(q+1,q+tot+1,cmp);
    memset(dis,233,sizeof(dis));
    dis[1]=0;
    for(int i=1;i<=tot;i++){
        if(q[i].x==n+1)break;
        if(!q[i].dis)ans[q[i].id]=dis[q[i].x];
        else dis[q[i].x]=max(dis[q[i].x],ans[q[i].id]+q[i].dis);
    }printf("%d\n",dis[T]<0?-1:t-dis[T]);
    return 0;
}
时间: 2024-10-09 16:48:18

BZOJ 1395 [Baltic2005]Trip(最短路+DP)的相关文章

BZOJ 1413 取石子游戏(DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1413 题意:n堆石子排成一排.每次只能在两侧的两堆中选择一堆拿.至少拿一个.谁不能操作谁输. 思路:参考这里. int f1[N][N],f2[N][N],n,a[N]; void deal() { RD(n); int i,j,k; FOR1(i,n) RD(a[i]),f1[i][i]=f2[i][i]=a[i]; int p,q,x; for(k=2;k<=n;k++) for(

BZOJ 2595 游览计划(插头DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2595 题意:给出一个数字矩阵.求一个连通块包含所有的数字0且连通块内所有数字之和最小. 思路:对于每个格子,是0则必须要选.那么对于不选的格子(i,j)在什么时候可以不选呢?必须同时满足以下两个条件: (1)(i,j)不是0: (2)(i-1,j)不选或者(i-1,j)选了但是轮廓线上还有别的地方与(i-1,j)是一个连通块. int Pre[105][N],op[105][N]; s

uva 10269 最短路+dp

题意:有a个村庄.b个城镇, 编号分别为:1-a , a+1--a+b . 有双神奇的鞋,可以瞬时移动,可以使用k次,每次可以移动L , 但穿这双鞋的时候,不能经过城镇 , 问:从a+b 到 1 最短距离是多少? 刚开始看这个题时 , 一点思路都没有 , dp类型的题目做得太少了. 解法:进行状态压缩, 用点+使用鞋子的次数 , 来表示一个状态 , d[i][k] , 表示到 点 i 使用 k 次鞋子的最短距离是多少. 但要先进行初始化 , 求任意点之间不经过城镇的最短距离 , 用floyd算法

BZOJ 1978 取数游戏(DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1978 题意:给出一个数列a,在其中找出下标依次增大的数,使得任意相邻的两个数的最大公约数大于等于m.找出最多的数字. 思路:f[i]表示前面的数字中最大公约数为i可以找出的最多的数字个数.那么对于当前数字x: 接着更新f: int f[N],a[N]; int n,m; int main() { RD(n,m); int i; FOR1(i,n) RD(a[i]); int j,k;

BZOJ 2436 Noi嘉年华(优化DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2436 题意:有一些活动,起始时间持续时间已知.有两个场地.每个活动最多只能在一个场地举行,且两个场地同一时间不能都举行活动.但是同一场地同一时间可以举行多个活动.要求的是两个场地中活动数目少的场地的活动数目的最大值S.再输出某个活动必须被安排时的S值. 思路:我直接粘贴原思路了. 区间离散化,设A={嘉年华1的活动}, B={嘉年华2的活动},C={未安排的活动}. 设num[i][j

BZOJ 1090 字符串折叠(区间DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1090 题意:字符串AAAAAAAAAABABABCCD的最短折叠为9(A)3(AB)CCD,注意数字的长度和圆括号都算最后长度.求一种折叠方式使得总长度最小. 思路:f[L][R]=min(R-L+1,f[L][i]+f[i+1][R]),另外若[L,R]能由[i+1,R]重复若干次,则也可用折叠后的长度更新f[L][R]. char s[N]; int f[N][N],n; int

BZOJ 1294 围豆豆Bean(DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1294 题意: 思路:f[i][j][st]表示从(i,j)出 发到(i,j)停止组成的回路.状态为st的最小步数.从每个0的位置(i,j)进行BFS一次,得到所有的状态.判断一个路径是否包含某个格子时,可以 从该格子向左发出一条射线,判断这条射线与路径交点个数.为奇数时包含. char s[N][N]; int f[N][N][1<<9]; int a[N],n,m,K; int b

[BZOJ 4033] [HAOI2015] T1 【树形DP】

题目链接:BZOJ - 4033 题目分析 使用树形DP,用 f[i][j] 表示在以 i 为根的子树,有 j 个黑点的最大权值. 这个权值指的是,这个子树内部的点对间距离的贡献,以及 i 和 Father[i] 之间的边对答案的贡献(比如这条边对黑点对距离和的贡献就是子树内部的黑点数 * 子树外部的黑点数 * 这条边的权值). 然后DFS来求,枚举 i 的每个儿子 j,现在的 f[i][] 是包含了 [1, j-1] 子树,然后两重循环枚举范围是 [1, j - 1] 的子树总 Size 和

[bzoj 1026]windy数(数位DP)

题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1026 分析: 简单的数位DP啦 f[i][j]表示数字有i位,最高位的数值为j的windy数总个数 那么f[i][j]=singma(f[i-1][k])(|j-k|>=2) 那么对于1~x(假设x从高到低的每位依次是x[n],x[n-1],……x[1])中的windy数个数就是f[n][0]+f[n][1]+……f[n][x[n]-1] + f[n-1][0]+f[n-1][1]+……f[