Divide and conquer:Telephone Lines(POJ 3662)

题目大意:一堆电话线要你接,现在有N个接口,总线已经在1端,要你想办法接到N端去,电话公司发好心免费送你几段不用拉网线,剩下的费用等于剩余最长电话线的长度,要你求出最小的费用。
这一看又是一个最小化最大值的问题(也可以看成是最大化最小值的问题),常规方法一样的就是把这个费用二分就好,但是这道题是道图论题,不一定经过所有的点,那我们就以二分基准长度为界限,把小于基准长度的那一部分看成是0,大于等于基准长度的看成是1,这样我们只用SPFA算法算最短路径就可以了,非常的巧妙
参考:http://poj.org/showmessage?message_id=181794
PS:好久没写SPFA了,都忘记是怎么写了,重新定义长度的时候又忘记乘以2了WA一个晚上真是日了
#include <iostream>
#include <algorithm>
#include <functional>
#define SIZE 1010 using namespace std;
typedef int Position; struct _set
{
Position ed;
int next;
int length;
}Path[];
struct _head
{
int point;
}Heads[SIZE];
static int dist[SIZE];
static bool visit[SIZE], oep[];
static Position que[(SIZE + ) * ]; void solve(const int, const int, const int, const int);
bool SPFA(const int, const int, const int, const int); int main(void)
{
int Sum_Poles, Free_Cables, Sum_Path, length, L_Max;
Position st, ed; while (~scanf("%d%d%d", &Sum_Poles, &Sum_Path, &Free_Cables))
{
L_Max = -;
for (int i = ; i <= Sum_Poles; i++)
Heads[i].point = -;
for (int i = ; i < * Sum_Path;)
{
scanf("%d%d%d", &st, &ed, &length);
//无向图,两边都要存
Path[i].ed = ed; Path[i].length = length; Path[i].next = Heads[st].point;
Heads[st].point = i++; Path[i].ed = st; Path[i].length = length; Path[i].next = Heads[ed].point;
Heads[ed].point = i++; L_Max = max(L_Max, length);
}
solve(Sum_Poles, Sum_Path, Free_Cables, L_Max);
}
return ;
} void solve(const int Sum_Poles, const int Sum_Path, const int Free_Cables, const int L_Max)
{
int lb = , rb = L_Max + , mid; while (rb - lb > )//对距离二分
{
mid = (lb + rb) >> ;
if (SPFA(mid, Sum_Path, Sum_Poles, Free_Cables)) lb = mid;
else rb = mid;
if (dist[Sum_Poles] == 0x3fffffff)
//任何一次寻找过后,如果图能到N点,那么N的dist值一定不是0x3fffffff
//否则,一定是不联通
{
printf("-1\n");
return;
}
}
printf("%d\n", lb);
} bool SPFA(const int x, const int Sum_Path, const int Sum_Poles, const int Free_Cables)
{
int head = , back = , out, to; que[head] = ; //开始是从1开始的 for (int i = ; i < * Sum_Path; i++)
oep[i] = Path[i].length < x ? : ; fill(dist, dist + Sum_Poles + , 0x3fffffff);
memset(visit, , sizeof(visit));
dist[] = ; while (head != back)
{
out = que[head]; head = (head + ) % ( * SIZE);
visit[out] = ; for (int k = Heads[out].point; k != -; k = Path[k].next)
{
to = Path[k].ed;
if (dist[out] + oep[k] < dist[to])
{
dist[to] = dist[out] + oep[k];
if (!visit[to])
{
visit[to] = ;
que[back] = to; back = (back + ) % ( * SIZE);
}
}
}
}
return dist[Sum_Poles] > Free_Cables;
}

Divide and conquer:Telephone Lines(POJ 3662)的更多相关文章
- Telephone Lines POJ - 3662 (二分+spfa)
Farmer John wants to set up a telephone line at his farm. Unfortunately, the phone company is uncoop ...
- Divide and conquer:K Best(POJ 3111)
挑选最美的珠宝 题目大意:挑选k个珠宝使得∑a/∑b最大,输出组合数 最大化平均值的标准题型,二分法就好了,一定要注意范围(10e-7),如果是10e-8就会tle,10e-6就是wa #inclu ...
- Divide and conquer:Dropping tests(POJ 2976)
最大化平均值 题目大意:给定你n个分数,从中找出k个数,使∑a/∑b的最大值 这一题同样的也可以用二分法来做(用DP会超时,可见二分法是多么的实用呵!),大体上是这样子:假设最大的平均值是w,那么题目 ...
- Divide and conquer:Aggressive Cows(POJ 2456)
侵略性的牛 题目大意:C头牛最大化他们的最短距离 常规题,二分法即可 #include <iostream> #include <algorithm> #include < ...
- Divide and Conquer:Cable Master(POJ 1064)
缆绳大师 题目大意,把若干线段分成K份,求最大能分多长 二分法模型,C(x)就是题干的意思,在while那里做下文章就可以了,因为这个题目没有要求长度是整数,所以我们要不断二分才行,一般50-100次 ...
- Divide and Conquer:Monthly Expense(POJ 3273)
Monthly Expense 题目大意:不废话,最小化最大值 还是直接套模板,不过这次要注意,是最小化最大值,而不是最大化最小值,判断的时候要注意 联动3258 #include <iostr ...
- Divide and Conquer:River Hopscotch(POJ 3258)
去掉石头 题目大意:一群牛在河上的石头上跳来跳去,现在问你如何通过去掉M个石头,使得牛跳过石头的最短距离变得最大? 这一题比较经典,分治法的经典,二分法可以很方便处理这个问题,我们只要明白比较函数这 ...
- POJ 3662 Telephone Lines【Dijkstra最短路+二分求解】
Telephone Lines Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 7214 Accepted: 2638 D ...
- (poj 3662) Telephone Lines 最短路+二分
题目链接:http://poj.org/problem?id=3662 Telephone Lines Time Limit: 1000MS Memory Limit: 65536K Total ...
随机推荐
- unity资源管理
Resources.Load(path); 每次执行都会真的去从硬盘加载资源,如果不希望这样做,那就保存第一次返回的引用,下次直接使用即可. Resources.UnloadAsset(obj); 该 ...
- 开机自动连接/断开VPN 批处理
或许大家在工作或生活中有接触到VPN,如果使用Windows自带的VPN来连接,每次开机要像宽带拨号那样,右击该VPN连接图标,然后选择“连接”(如果未记住密码甚至还要输入密码),然后点击确定,有点麻 ...
- runtime(面试)
运行时机制,runtime库里面包含了跟类.成员变量.方法相关的API,比如获取类里面的所有成员变量,为类动态添加成员变量,动态改变类的方法实现,为类动态添加新的方法等 需要导入<objc/me ...
- 4. Linux常用命令
1. ls 查看当前目录信息 2. pwd 查看当前目录 3. cd 切换目录 快捷操作:cd - 可快速对最近的两个目录切换, cd 或者cd~ 直接回到用户自己的主目录, 4. hwclock ...
- BZOJ3083——遥远的国度
1.题目大意:三个操作,换根,修改树上的某条路径,查询一个子树的最小值 2.分析:这个其实还是挺好做的,修改树上的某条路径,裸树剖,查询子树的最小值,这个是树剖满足dfs序 那么就是换根了,对吧,其实 ...
- 热更新脚本C#light,ulua,Scorpio性能比较
http://www.unity蛮牛.com/thread-32861-1-1.html 测试环境: unity4.5.2 三个脚本全是源码导入 PC :处理器 Intel(R) Core(TM) ...
- NGUI Sprite 和 Label 改变Layer 或父物体后 未更新深度问题
using UnityEngine; using System.Collections.Generic; /// <summary> /// Sprite is a textured el ...
- cocos2d-x 内存管理浅析
Cocos2d-x用create创建对象, 这个方法已经被引擎封装成一个宏定义了:CREATE_FUNC, 下面是这个宏定义的实现: #define CREATE_FUNC(__TYPE__) \ ...
- (原创)用Receiver和SystemService监听网络状态,注册Receiver的两种方式
android中网络编程不可避免地要监听网络状态,wifi或者3G以及以太网,并根据当前状态做出相应决策. 在MyReceiver类中的onReceive方法中获得系统服务 ConnectivityM ...
- linux回收站设计
linux回收站设计 在windows下有一个很好的东西,那就是回收站,虽然有很多人批评它.linux不是没有回收站,很多桌面环境都可以看到是有回收站的. 这里是讨论如何设计一个回收站,而不是有没有的 ...