poj_3662 最小化第k大的值
题目大意
有N个节点以及连接的P个无向边,现在要通过这P条边从1号节点连接到N号节点。若无法连接成功,则返回-1;若能够连接成功,那么其中用到了L条边,这L条边中有K条边可以免费,L-K条边不能免费,求出不能免费的边的最大长度。
题目分析
判断能否到达,可以通过BFS搜索路径,若不能到达,返回-1;若能到达,且最少需要的路径的边数小于等于K,那么所有的边都可以免费,则返回0;若能够到达,且最少需要的路径边数大于K,则需要求出从节点1到节点N的路径中第K+1长的边的最小值,即最小化第k大的值问题
。
用二分法,枚举边长x,若路径中大于等于x的边数大于K,则说明x在构成路径的边长序列中的序号大于K+1,则将x增大;若路径中大于等于x的边长小于等于K,则说明x在构成路径的边长序列中的序号小于等于K+1,因此将x减小.... 直到x为最小的满足路径中边长大于等于x的边数大于K的最小值
,则x即为路径中第K+1大的边,且X最小。
那么,对于一个长度x,该选择哪条路径呢?由于要求的最小的x,那么就要求路径中边长大于等于x的个数尽可能的少(若个数少于等于K,则减小x),转换一下,将路径中长度大于等于x的边权值设为1,小于x的权值设为0,则走一条从源到汇的路径,路径中边的权值和最小即对应路径中边长大于等于x的个数最少。
求最短路径,使用Dijkstra算法即可。
实现(c++)
#include<stdio.h>
#include<string.h>
#include<vector>
#include<queue>
#include<algorithm>
using namespace std;
#define MAX_NODE 1005
#define MAX_EDGE 20005
#define INF 1 << 28
#define min(a, b) a < b?a:b
#define max(a, b) a >b? a:b
struct Edge{
int to;
int d;
int next;
};
struct NodeDist{
int v;
int d;
NodeDist(int vv = 0, int dd = 0) :
v(vv), d(dd){};
};
struct Cmp{
bool operator()(const NodeDist& e1, const NodeDist& e2){
return e1.d > e2.d;
}
};
Edge gEdges[MAX_EDGE];
int gEdgeCount;
int gHead[MAX_NODE];
int gDist[MAX_NODE];
bool gVisited[MAX_NODE];
void InsertEdge(int u, int v, int d){
int e = gEdgeCount++;
gEdges[e].to = v;
gEdges[e].d = d;
gEdges[e].next = gHead[u];
gHead[u] = e; e = gEdgeCount++;
gEdges[e].to = u;
gEdges[e].d = d;
gEdges[e].next = gHead[v];
gHead[v] = e;
} int Dijkstra(int s, int t, int k){
memset(gVisited, false, sizeof(gVisited));
memset(gDist, 0x7F, sizeof(gDist));
gDist[s] = 0;
priority_queue<NodeDist, vector<NodeDist>, Cmp> pq;
pq.push(NodeDist(s, 0));
NodeDist nd;
while (!pq.empty()){
nd = pq.top();
pq.pop();
if (gVisited[nd.v])
continue;
gVisited[nd.v] = true;
if (nd.v == t){
break;
}
for (int e = gHead[nd.v]; e != -1; e = gEdges[e].next){
int v = gEdges[e].to;
if (gDist[v] > gDist[nd.v] + (gEdges[e].d >= k)){
gDist[v] = gDist[nd.v] + (gEdges[e].d >= k);
pq.push(NodeDist(v, gDist[v]));
}
}
}
return gDist[t];
} int MinStep(int s, int t){
bool visited[MAX_NODE];
memset(visited, false, sizeof(visited));
queue<pair<int, int> > Q;
Q.push(pair<int, int>(s, 0));
visited[s] = true;
while (!Q.empty()){
pair<int, int> p = Q.front();
Q.pop();
if (p.first == t){
return p.second;
}
for (int e = gHead[p.first]; e != -1; e = gEdges[e].next){
if (!visited[gEdges[e].to]){
Q.push(pair<int, int>(gEdges[e].to, p.second + 1));
visited[gEdges[e].to] = true;
}
}
}
return -1;
} int gEdgeDist[MAX_EDGE];
int main(){
int n, p, k, u, v, d, e_count;
while (scanf("%d %d %d", &n, &p, &k) != EOF){
memset(gHead, -1, sizeof(gHead));
gEdgeCount = 0;
e_count = 0;
for (int i = 0; i < p; i++){
scanf("%d %d %d", &u, &v, &d);
InsertEdge(u, v, d);
gEdgeDist[e_count++] = d;
} int min_step = MinStep(1, n);
if (min_step == -1){
printf("-1\n");
continue;
}
else if (min_step <= k){
printf("0\n");
continue;
} sort(gEdgeDist, gEdgeDist + e_count);
int beg = 0, end = e_count, mid;
int rr;
while (beg < end){
mid = (beg + end) / 2;
int t = Dijkstra(1, n, gEdgeDist[mid]);
if (t <= k){
end = mid;
}
else{
rr = gEdgeDist[mid];
beg = mid + 1;
}
}
//最后得到的是,满足路径中边长大于等于 x 的长度的边数 大于k 最大的 x printf("%d\n", rr);
}
return 0;
}
poj_3662 最小化第k大的值的更多相关文章
- POJ 3662 Telephone Lines (二分+Dijkstra: 最小化第k大的值)
题意 Farmer John想从电话公司修一些电缆连接到他农场.已知N个电线杆编号为1,2,⋯N,其中1号已经连接电话公司,N号为农场,有P对电线杆可连接. 现给出P对电线杆距离Ai,Bi,Li表示A ...
- POJ-3662 Telephone Lines---二分+最短路+最小化第k+1大
题目链接: https://cn.vjudge.net/problem/POJ-3662 题目大意: 求一条路径从1到n使第k+1大的边最小. 解题思路: 二分答案mid,当原边权小于等于mid新边权 ...
- 利用划分树求解整数区间内第K大的值
如何快速求出(在log2n的时间复杂度内)整数区间[x,y]中第k大的值(x<=k<=y)? 其实我刚开始想的是用快排来查找,但是其实这样是不行的,因为会破坏原序列,就算另外一个数组来存储 ...
- 查找第K大的值
这种题一般是给定N个数,然后N个数之间通过某种计算得到了新的数列,求这新的数列的第K大的值 POJ3579 题意: 用$N$个数的序列$x[i]$,生成一个新序列$b$. 新的序列定义为:对于任意的$ ...
- 算法---数组总结篇2——找丢失的数,找最大最小,前k大,第k小的数
一.如何找出数组中丢失的数 题目描述:给定一个由n-1个整数组成的未排序的数组序列,其原始都是1到n中的不同的整数,请写出一个寻找数组序列中缺失整数的线性时间算法 方法1:累加求和 时间复杂度是O(N ...
- 两个已排序数组进行合并后的第K大的值--进军硅谷
我看到此题时,首先想到一个一个比较遍历过去,这是最暴力的方法,后面我想到了已经排序,那么对每个数组进行二分,然后比较这两个值.此书第三种解法,挺不错,只对那个长度较小的数组进行二分查找,保证i+j-1 ...
- POJ_3579_Median_(二分,查找第k大的值)
描述 http://poj.org/problem?id=3579 给你一串数,共C(n,2)个差值(绝对值),求差值从大到小排序的中值,偶数向下取. Median Time Limit: 1000M ...
- POJ 3579 3685(二分-查找第k大的值)
POJ 3579 题意 双重二分搜索:对列数X计算∣Xi – Xj∣组成新数列的中位数 思路 对X排序后,与X_i的差大于mid(也就是某个数大于X_i + mid)的那些数的个数如果小于N / 2的 ...
- 215. Kth Largest Element in an Array找出数组中第k大的值
堆排序做的,没有全部排序,找到第k个就结束 public int findKthLargest(int[] nums, int k) { int num = 0; if (nums.length &l ...
随机推荐
- 枚举类转成json
import com.alibaba.fastjson.JSONArray; import com.alibaba.fastjson.JSONObject; /** * portlet类别枚举类 */ ...
- CentOs6.5 安装rabbitmq(转)
// 安装预环境 yum install gcc gcc-c++ yum install zlib zlin-devel ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 / ...
- 文件服务器和web应用分离的思路(转)
目前在做一个应用,有不同的客户端,包括web应用的客户端,其他的客户端,都要访问我的文件服务器,通过文件服务程序提供的服务来访问文件,但是对文件管理服务器这个应用,没有什么思路,请大家给点思路,谢谢: ...
- PHP程序员的10个有用的技巧和教程
PHP被定义为一个通用的服务器端脚本语言,它基本上是专为Web开发的重要目的.借助PHP可以创建动态和惊人的网页效果,他是被嵌入在一个HTML源文件的服务器端的脚本语言之一.因此,它成为最流行的,也是 ...
- sql server merge 的用法
CREATE TABLE tTable ( id INT , f1 VARCHAR(10) , f2 VARCHAR(10) , f3 VARCHAR(10) ) GO INSERT INTO tTa ...
- Qt 反射
简介 本文主要讲解Qt是如何实现反射,以及一点点反射使用的小心得. 文章概览 Qt反射内幕小窥 详细内容 反射前期准备 得到注册的类成员变量 得到注册的类成员函数 访问类成员属性(get,set) 调 ...
- altium designer应用技巧---cyclone IV代芯片底部焊盘问题
首先对于 altera 公司的FPGA芯片来讲,在cyclone III代以上,芯片的底部增加了一 个焊盘,很多工程师往往以为是散热用,其实不然,底部焊盘需要接地(altera手册上面 明确规定,Th ...
- 【转】C# Async/Await 异步编程中的最佳做法
Async/Await 异步编程中的最佳做法 Stephen Cleary 近日来,涌现了许多关于 Microsoft .NET Framework 4.5 中新增了对 async 和 await 支 ...
- HTC Desire 816 root教程和方法
每个手机入手之后基本上都需要进行root,不root的话,手机里很多的无有软件都删除不了,咱们的HTC Desire 816也是一样的,也需要进行root才可以删除系统里自带的那些无用的软件,这些软件 ...
- CentOS基础命令大全
1.关机 (系统的关机.重启以及登出 ) 的命令 shutdown -h now 关闭系统(1) init 0 关闭系统(2) telinit 0 关闭系统(3) shutdown -h hours: ...