题目描述

FarmerJohn打算将电话线引到自己的农场,但电信公司并不打算为他提供免费服务。于是,FJ必须为此向电信公司支付一定的费用。FJ的农场周围分布着N(1<=N<=1,000)根按1..N顺次编号的废弃的电话线杆,任意两根电话线杆间都没有电话线相连。一共P(1<=P<=10,000)对电话线杆间可以拉电话线,其余的那些由于隔得太远而无法被连接。第i对电话线杆的两个端点分别为A_i、B_i,它们间的距离为L_i(1<=L_i<=1,000,000)。数据中保证每对{A_i,B_i}最多只出现1次。编号为1的电话线杆已经接入了全国的电话网络,整个农场的电话线全都连到了编号为N的电话线杆上。也就是说,FJ的任务仅仅是找一条将1号和N号电话线杆连起来的路径,其余的电话线杆并不一定要连入电话网络。经过谈判,电信公司最终同意免费为FJ连结K(0<=K<N)对由FJ指定的电话线杆。对于此外的那些电话线,FJ需要为它们付的费用,等于其中最长的电话线的长度(每根电话线仅连结一对电话线杆)。如果需要连结的电话线杆不超过K对,那么FJ的总支出为0。请你计算一下,FJ最少需要在电话线上花多少钱。

输入格式

第1行: 3个用空格隔开的整数:N,P,以及K

第2..P+1行: 第i+1行为3个用空格隔开的整数:A_i,B_i,L_i

输出格式

第1行: 输出1个整数,为FJ在这项工程上的最小支出。

如果任务不可能完成, 输出-1


容易想到的策略是:把1~n路径上花费最高的k条路径去掉。所以我们可以枚举每条路径并枚举路径上的每条边来解答。但这显然不是我们能够接受的复杂度。

我们需要换一种思路来做。我们来分析一下答案的性质:

设答案为ans,那么在1~n的所有路径中lenth<=ans的都可以免费,其余的计入k条免费线路中。如果ans过小,那么lenth>ans的路径数可能大于k,如果过大则得不到最优解。所以我们可以二分答案,对于每个二分出的limitation,把边长大于它的路线的代价改成1,其余的代价为0。然后跑一遍最短路,看最短路长度是否≤k即可。

最短路可以用Dijkstra+Heap做到O((N+M) * logN),二分答案的复杂度为O(logAns),所以总复杂度为:

\[ O((N+M)logNlogAns) \]

可以通过本题。

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<queue>
#define maxn 1001
#define maxm 10001
using namespace std;struct edge{int to,next,dis;edge(){}edge(const int &_to,const int &_dis,const int _next){to=_to,dis=_dis,next=_next;}
}e[maxm<<1];
int head[maxn],k;int dis[maxn],maxlen,ans;
bool vis[maxn];
int n,m,t;
priority_queue< pair<int,int>,vector< pair<int,int> >,greater< pair<int,int> > > q;inline int read(){register int x(0),f(1); register char c(getchar());while(c<'0'||'9'<c){ if(c=='-') f=-1; c=getchar(); }while('0'<=c&&c<='9') x=(x<<1)+(x<<3)+(c^48),c=getchar();return x*f;
}
inline void add(const int &u,const int &v,const int &w){e[k]=edge(v,w,head[u]);head[u]=k++;
}inline void dijkstra(const int &lim){memset(dis,0x3f,sizeof dis);memset(vis,false,sizeof vis);q.push(make_pair(0,1));dis[1]=0;while(q.size()){int u=q.top().second; q.pop();if(vis[u]) continue; vis[u]=true;for(register int i=head[u];~i;i=e[i].next){int v=e[i].to;if(dis[v]>dis[u]+(e[i].dis>lim))dis[v]=dis[u]+(e[i].dis>lim),q.push(make_pair(dis[v],v));}}
}int main(){memset(head,-1,sizeof head);n=read(),m=read(),t=read();for(register int i=1;i<=m;i++){int u=read(),v=read(),w=read();add(u,v,w),add(v,u,w);if(maxlen<w) maxlen=w;}int l=0,r=maxlen,flag=false;while(l<=r){int lim=l+r>>1;dijkstra(lim);if(dis[n]<=t) flag=true,ans=lim,r=lim-1;else l=lim+1;}printf("%d\n",flag?ans:-1);return 0;
}

转载于:https://www.cnblogs.com/akura/p/10848893.html

[Usaco2007 Jan]Telephone Lines架设电话线相关推荐

  1. [BZOJ] 1614: [Usaco2007 Jan]Telephone Lines架设电话线

    1614: [Usaco2007 Jan]Telephone Lines架设电话线 Time Limit: 5 Sec  Memory Limit: 64 MB Submit: 1806  Solve ...

  2. bzoj 1614: [Usaco2007 Jan]Telephone Lines架设电话线(二分+SPFA)

    1614: [Usaco2007 Jan]Telephone Lines架设电话线 Time Limit: 5 Sec  Memory Limit: 64 MB Submit: 1761  Solve ...

  3. [BZOJ1614][Usaco2007 Jan]Telephone Lines架设电话线

    [Usaco2007 Jan]Telephone Lines架设电话线 时间限制: 1 Sec 内存限制: 128 MB 题目描述 Farmer John打算将电话线引到自己的农场,但电信公司并不打算 ...

  4. BZOJ——1614: [Usaco2007 Jan]Telephone Lines架设电话线

    Time Limit: 5 Sec  Memory Limit: 64 MB Submit: 1930  Solved: 823 [Submit][Status][Discuss] Descripti ...

  5. 【二分答案】【最短路】bzoj1614 [Usaco2007 Jan]Telephone Lines架设电话线

    对于二分出的答案x而言,验证答案等价于将所有边权>x的边赋成1,否则赋成0,然后判断从1到n的最短路是否<=K. #include<cstdio> #include<cs ...

  6. [bzoj1614]: [Usaco2007 Jan]Telephone Lines架设电话线

    传送门 题意:给一个图,定义两点间的距离为路径上最大的边权,可以将路径上不多于k条边的权值变为0,求两点间最小距离 二分答案,判断时只要将大于当前二分值的边记为1,否则记为0,做一遍spfa,判断di ...

  7. bzoj1705[Usaco2007 Nov]Telephone Wire 架设电话线(dp优化)

    1705: [Usaco2007 Nov]Telephone Wire 架设电话线 Time Limit: 5 Sec  Memory Limit: 64 MB Submit: 441  Solved ...

  8. 【bzoj1705】[Usaco2007 Nov]Telephone Wire 架设电话线 dp

    题目描述 最近,Farmer John的奶牛们越来越不满于牛棚里一塌糊涂的电话服务 于是,她们要求FJ把那些老旧的电话线换成性能更好的新电话线. 新的电话线架设在已有的N(2 <= N < ...

  9. bzoj 1614 Telephone Lines架设电话线 - 二分答案 - 最短路

    Description Farmer John打算将电话线引到自己的农场,但电信公司并不打算为他提供免费服务.于是,FJ必须为此向电信公司支付一定的费用. FJ的农场周围分布着N(1 <= N ...

最新文章

  1. MySQL 调用存储过程
  2. Python这些位运算的妙用,绝对让你大开眼界
  3. 使用ffmpeg 的 filter 给图片添加水印
  4. 基于Ubuntu Server 16.04 LTS版本安装和部署Django之(二):Apache安装和配置
  5. OMP: Error #15: Initializing libiomp5md.dll, but found libiomp5md.dll already initialized.
  6. VS2012下基于Glut 绘制立方体示例程序:
  7. 伯克利区块链课程:替代密码验证,比特币应用数字签名的进化之路
  8. windows系统TLQ8安装时提示载入java vm时windows出现错误
  9. 阿里面试整个流程(已OC)
  10. mc小刘yeah粉丝网
  11. 网络综合布线应用指南
  12. 【详细】Python实现病毒仿真器
  13. 中科大计算机学院推免拟录取,中科大2019年录取177所高校推免生2109人,外校985生源不足三成...
  14. 让linux识别html,8 款浏览器对 HTML5 支持评测
  15. C# 语音端点检测(VAD)实现过程分析
  16. Android系统USB读卡器
  17. idea提示java.sql.SQLException: Access denied for user ‘‘@‘localhost‘ (using password: NO)
  18. 畅玩三子棋(可选择棋盘大小)
  19. TWS耳机行业现状:苹果想通吃,微软欲入局
  20. 智慧城市水质在线蓝绿藻监测传感器

热门文章

  1. Android 极广推送接入
  2. android Wifi开发相关内容
  3. android资源的热更新(替换 AssetManager+LoadedApk中的资源路径)
  4. Android 蓝牙操作--读取远程已配对的蓝牙设备
  5. Android 应用程序发布流程注意事项(整理)
  6. RabbitMQ—队列迁移插件shovel的使用
  7. c++学习笔记之输入/输出流
  8. kotlin学习笔记——扩展函数(anko)和网络请求
  9. 暑假集训考试反思+其它乱写
  10. Jmeter当获取正则表达式匹配数字为负数时获取所有匹配的值