题意描述

[BOI 2002]双调路径

题意描述的确实不是很清楚(出题人惜字如金)。

给定一张有 \(n\) 个点,\(m\) 条边的无向图,每条边有两个权值,分别表示经过这个点的代价和时间。

同时给出起点 \(s\) 和终点 \(t\),显然 \(s\to t\) 的路径有很多条。

其中说路径 A 比路径 B 更优,当且仅当 A 的代价和时间都小于 B。

当没有路线比这条路线更优时,称这条路线为最优路线。题目要求求 \(s\to t\) 的最优路线的条数。

注意:这里的最优路线可以有很多条,如果 \(time_A>time_B\) 但是 \(price_A<price_B\) 则两者谁也不优于谁。

同时,当两条路线代价和时间都相等时,成这两条路线为一条。(这里很良心了)

算法分析

基本思路

这里是双权值的最短路问题,比较直接的方法是以代价为关键字来枚举路径,每次选取时间最小的路径即可。

但是可以发现这种暴力算法将耗费大量时间用于判断代价是否相同,时间复杂度将非常恐怖,所以当场枪毙。

当时这种思想也给了我们启发,一番整理后有了一个可行的思想:

令 \(dis(i,j)\) 表示从 \(s\) 到 \(i\) 的代价为 \(j\) 的最短时间,那么显然有递推式:

\[dis(i,j)=min_{k\to i\in E}\{dis(k,j-price_{k,i})+time_{k,i}\}
\]

统计答案时,我们一次遍历 \(dis(t,i)(0\leq i\leq 10000)\)。(其中 10000 为数据范围内的最大代价)

同时维护一个 \(mn\) 表示当前时间的最小值,当 \(dis(t,i)<mn\) 时,更新 \(mn\) 并且令 ans++

这就表示:当一条路径的代价比你大(从小到大枚举)但时间比你少时,这是最优路径之一。

然后再时限放宽之后应该就是可以过了。(2020.4.22)

时间优化

但是在之前,这还是一道紫题的时候...,它的实现是 0.1s,这么“滥用”最短路肯定是过不去的。

那怎么办呢?

可以发现,根据 OI 定理:当一个人比你小还比你强时,你就可以退役了

换句话说:如果在你之前存在一条路径,代价和时间都比你小,那么你不可能是最优路径。

不仅如此,你所拓展的路径也不可能是最优路径。(自证不难)

所以我们可以利用树状数组进行优化,记录下区间最小值即可。(记得是二位树状数组)

代码实现

实现比较简单,说实话思路也没有难到哪里去。

#include<cstdio>
#include<cstring>
#include<iostream>
#include<cmath>
#include<queue>
#include<algorithm>
#define N 1100
#define M 3100
#define C 10010
using namespace std; int n,m,s,t,head[N],cnt=0,dis[N][C],tree[N][C];
bool vis[N][C];
struct Edge{
int nxt,to,w1,w2;
}ed[M<<1]; int read(){
int x=0,f=1;char c=getchar();
while(c<'0' || c>'9') f=(c=='-')?-1:1,c=getchar();
while(c>='0' && c<='9') x=x*10+c-48,c=getchar();
return x*f;
} void add(int u,int v,int c,int t){
ed[++cnt].nxt=head[u];
ed[cnt].to=v,ed[cnt].w1=c,ed[cnt].w2=t;
head[u]=cnt;
return;
} int lowbit(int x){return x&(-x);} void change(int x,int y,int z){
y++;
while(y<10100){
tree[x][y]=min(tree[x][y],z);
y+=lowbit(y);
}
return;
} int ask(int x,int y){
y++;
int mn=0x7fffffff;
while(y>=1){
mn=min(mn,tree[x][y]);
y-=lowbit(y);
}
return mn;
} void spfa(){
queue<pair<int,int> >q;
memset(dis,0x3f,sizeof(dis));
memset(tree,0x3f,sizeof(tree));
memset(vis,false,sizeof(vis));
dis[s][0]=0;
vis[s][0]=true;
q.push(make_pair(s,0));
change(s,0,0);
while(!q.empty()){
int w=q.front().second,u=q.front().first;
q.pop();
vis[u][w]=false;
for(int i=head[u];i;i=ed[i].nxt){
int v=ed[i].to;
int w1=ed[i].w1,w2=ed[i].w2;
int ww=w+w1;
if(ask(v,ww)>dis[u][w]+w2){
dis[v][ww]=dis[u][w]+w2;
change(v,ww,dis[v][ww]);
if(!vis[v][ww])
q.push(make_pair(v,ww)),vis[v][ww]=true;
}
}
}
return;
} int main(){
n=read(),m=read(),s=read(),t=read();
int u,v,w1,w2;
for(int i=1;i<=m;i++){
u=read(),v=read(),w1=read(),w2=read();
add(u,v,w1,w2),add(v,u,w1,w2);
}
spfa();
int mmin = 0x3f3f3f3f, ans = 0;
for (int i = 0; i <= 10000; i++) {
if (dis[t][i] < mmin)
mmin = dis[t][i], ans++;
}
printf("%d\n",ans);
return 0;
}

完结撒花

P5530 [BOI 2002]双调路径的更多相关文章

  1. 题解 P5530 [BalticOI 2002]双调路径

    P5530 [BalticOI 2002]双调路径 输入样例: 4 5 1 4 2 1 2 1 3 4 3 1 2 3 1 2 3 1 1 4 2 4 2 4 样例如下图 样例说明: 从1到4有4条路 ...

  2. Bicriterial routing 双调路径 HYSBZ - 1375(分层最短路)

    Description 来越多,因此选择最佳路径是很现实的问题.城市的道路是双向的,每条道路有固定的旅行时间以及需要支付的费用.路径由连续的道路组成.总时间是各条道路旅行时间的和,总费用是各条道路所支 ...

  3. bzoj1375 双调路径

    Description 来越多,因此选择最佳路径是很现实的问题.城市的道路是双向的,每条道路有固定的旅行时间以及需要支付的费用.路径由连续的道路组成.总时间是各条道路旅行时间的和,总费用是各条道路所支 ...

  4. [bzoj1375] [Baltic2002] Bicriterial routing 双调路径

    Description 如今的道路收费发展很快.道路的密度越来越大,因此选择最佳路径是很现实的问题.城市的道路是双向的,每条道路有固定的旅行时间以及需要支付的费用. 路径是连续经过的道路组成的.总时间 ...

  5. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  6. 8.2/baltic神(水)题

    summary:10 bzoj1334: Description N个政党要组成一个联合内阁,每个党都有自己的席位数. 现在希望你找出一种方案,你选中的党的席位数要大于总数的一半,并且联合内阁的席位数 ...

  7. POJ 2677 Tour

    题意:双调欧几里得旅行商问题.算法导论15-1题,从最左边的点严格从左走到右再从右走到左回到起点,所有点都要走且只走一次,求最短路径. 解法:定义dp[i][j]表示从i走到j的双调路径,分为两种情况 ...

  8. loj题目总览

    --DavidJing提供技术支持 现将今年7月份之前必须刷完的题目列举 完成度[23/34] [178/250] 第 1 章 贪心算法 √ [11/11] #10000 「一本通 1.1 例 1」活 ...

  9. 【正睿oi省选十连测】第一场

    四小时写了两个暴力??自闭 [原来这就是神仙们的分量Orz rank 56/75 可以说是无比垃圾了 下周目标:进步十名?[大雾 T1 题意:有n个点的图 点有点权Ai 也有点权Bi = A_1 + ...

随机推荐

  1. Go-err is shadowed during return

    where? Go程序函数中在通过 return关键字返回的时候,报错 why? 变量作用域的问题,在子作用域定义一个上层作用域的同名的变量 package main import ( "e ...

  2. newifi3-D2 openwrt挂载u盘扩容/overlay

    格式化U盘 1.openwrt安装插件 opkg install fdisk swap-utils kmod-usb-storage kmod-fs-ext4 e2fsprogs kmod-usb-o ...

  3. 三级菜单打怪升级,young -> plus -> pro

    young -> simple 三级菜单超简单,每层都是小循环 小可爱,不是if就是for,真可爱 def menu(message): print('按q返回上一层') print('按e退出 ...

  4. 18-SE-你说的都队

    文章目录 前言 建设银行app分析 招商银行app分析 中国银行app分析 工商银行app分析 总结 团队成员分工与评分 前言 18-SE-你说的都队所选项目题目为"村镇银行储蓄业务系统开发 ...

  5. C语言中 malloc

    参考:https://blog.csdn.net/kokodudu/article/details/11760863 一.malloc()和free()的基本概念以及基本用法: 1.函数原型及说明: ...

  6. 手写一个类SpringBoot的HTTP框架:几十行代码基于Netty搭建一个 HTTP Server

    本文已经收录进 : https://github.com/Snailclimb/netty-practical-tutorial (Netty 从入门到实战:手写 HTTP Server+RPC 框架 ...

  7. SpringBoot+Activiti+bpmn.js+Vue.js+Elementui(OA系统审批流)

    引言:OA系统用到请假.加班.调休.离职,需要使用工作流进行流程审批 一:activiti流程设计器的选择(通过学习activiti工作流过程中,发现一款好的流程设计器将会更好的方便的设计好流程(主要 ...

  8. day44 Pyhton 数据库Mysql

    内容回顾 什么是进程? 就是为了形容执行中的程序的一种称呼 它是操作系统中资源分配的最小单位 进程之间是数据隔离的,占用操作系统资源相对多 独立存在的 谈谈你对并发的理解 同时有多个任务需要执行,但是 ...

  9. swoft 事件监听和触发 打印sql日志

    需求 打印出swoft的所有sql日志到控制台或者文件 只要打开listener 下面 Dbranlisten.php 里面最后一行注释即可,swoft已经帮我们实现好了 ____ _____ ___ ...

  10. python 爬取链家

    import json import requests from lxml import etree from time import sleep url = "https://sz.lia ...