POJ 2728 Desert King(最优比率生成树, 01分数规划)
题意:
给定n个村子的坐标(x,y)和高度z, 求出修n-1条路连通所有村子, 并且让 修路花费/修路长度 最少的值
两个村子修一条路, 修路花费 = abs(高度差), 修路长度 = 欧氏距离
分析:
01分数划分的题目, 构造出 d[i] = 修路花费 - L * 修路长度, 这个L值我们可以二分(这道题看数据范围的话二分上限其实挺大的, 但其实上限取到100就可以过), 也可以用Dinkelbach迭代出来。
二分(1422ms)
#include <stdio.h>
#include <string.h>
#include <iostream>
#include <iostream>
#include <algorithm>
#include <vector>
#include <queue>
#include <set>
#include <map>
#include <cstring>
#include <cmath>
#define rep(i,a,b) for(int i = a; i < b;i++)
#define _rep(i,a,b) for(int i = a; i <= b;i++)
using namespace std;
const int maxn = + ;
const double inf = 1e9 + ;
const double eps = 1e-;
int n; double G[maxn][maxn];
double x[maxn], y[maxn], z[maxn]; inline double p2p_dis(double x1, double y1, double x2, double y2){
return sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
}
double prim(double L){
double tree_dis = ;
double dis[maxn];
int vis[maxn];
fill(dis, dis + maxn, inf);
memset(vis, ,sizeof(vis)); dis[] = ;
vis[] = ; for(int i = ; i < n; i++) dis[i] = abs(z[] - z[i]) - L * G[][i]; for(int times = ; times < n - ; times++){
int picked = -;
double min_dis = inf;
for(int i = ; i < n; i++){
if(!vis[i] && dis[i] < min_dis){
min_dis = dis[i];
picked = i;
}
}
tree_dis += min_dis;
vis[picked] = ;
for(int i = ; i < n; i++)
if(!vis[i])
dis[i] = min(dis[i] , abs(z[picked] - z[i]) - L * G[picked][i]) ;
}
return tree_dis;
} int main(){
// freopen("1.txt","r", stdin);
while(cin >> n && n){
rep(i,,n) cin >> x[i] >> y[i] >> z[i];
rep(i,,n) rep(j,i+,n) G[i][j] = G[j][i] = p2p_dis(x[i],y[i],x[j],y[j]);
double l = , r = ;
while(abs(r - l) > eps){
double mid = (l + r) / ;
// printf("%.3f\n", mid);
if(prim(mid) >= ){
l = mid;
}else{
r = mid;
}
}
printf("%.3f\n", l);
}
return ;
}
Dinkelbach(204ms)
#include <stdio.h>
#include <string.h>
#include <iostream>
#include <iostream>
#include <algorithm>
#include <vector>
#include <queue>
#include <set>
#include <map>
#include <cstring>
#include <cmath>
#define rep(i,a,b) for(int i = a; i < b;i++)
#define _rep(i,a,b) for(int i = a; i <= b;i++)
using namespace std;
const int maxn = + ;
const double inf = 1e9 + ;
const double eps = 1e-;
int n; double G[maxn][maxn];
double x[maxn], y[maxn], z[maxn]; inline double p2p_dis(double x1, double y1, double x2, double y2){
return sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
}
double prim(double L){
double ele = , deno = ; //分子分母
double dis[maxn];
int near[maxn]; //开一个数组记录每次选中是哪一条边 int vis[maxn];
fill(dis, dis + maxn, inf);
memset(vis, ,sizeof(vis));
memset(near, -, sizeof(near));
dis[] = ;
vis[] = ; for(int i = ; i < n; i++) dis[i] = abs(z[] - z[i]) - L * G[][i], near[i] = ; //每个点都是由0点更新的, 所以near都是0 for(int times = ; times < n - ; times++){
int picked = -;
double min_dis = inf;
for(int i = ; i < n; i++){
if(!vis[i] && dis[i] < min_dis){
min_dis = dis[i];
picked = i;
}
}
ele += abs(z[near[picked]] - z[picked]);//分子是选的路程
deno += G[near[picked]][picked]; //分母是真实的花费
vis[picked] = ;
for(int i = ; i < n; i++)
if(!vis[i] && dis[i] > abs(z[picked] - z[i]) - L * G[picked][i]){
dis[i] = abs(z[picked] - z[i]) - L * G[picked][i];
near[i] = picked;//如果通过picked点更新dis,那么该位置near就是picked
} } return ele/deno; //注意返回的跟二分不一样, 应该返回当前L下真正题目的所求
} int main(){
// freopen("1.txt","r", stdin);
while(cin >> n && n){
rep(i,,n) cin >> x[i] >> y[i] >> z[i];
rep(i,,n) rep(j,i+,n) G[i][j] = G[j][i] = p2p_dis(x[i],y[i],x[j],y[j]);
double L, ans = ;
while(){
L = ans;
ans = prim(L); //这是一个真实值, 不是含L的变式
if(abs(ans - L)< eps) break;
}
printf("%.3f\n", ans);
}
return ;
}
POJ 2728 Desert King(最优比率生成树, 01分数规划)的更多相关文章
- POJ 2728 Desert King 最优比率生成树
Desert King Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 20978 Accepted: 5898 [Des ...
- POJ.2728.Desert King(最优比率生成树 Prim 01分数规划 二分/Dinkelbach迭代)
题目链接 \(Description\) 将n个村庄连成一棵树,村之间的距离为两村的欧几里得距离,村之间的花费为海拔z的差,求花费和与长度和的最小比值 \(Solution\) 二分,假设mid为可行 ...
- POJ 2728 Desert King(最优比率生成树 01分数规划)
http://poj.org/problem?id=2728 题意: 在这么一个图中求一棵生成树,这棵树的单位长度的花费最小是多少? 思路: 最优比率生成树,也就是01分数规划,二分答案即可,题目很简 ...
- Desert King (poj 2728 最优比率生成树 0-1分数规划)
Language: Default Desert King Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 22113 A ...
- POJ 2728 Desert King (最优比率树)
题意:有n个村庄,村庄在不同坐标和海拔,现在要对所有村庄供水,只要两个村庄之间有一条路即可,建造水管距离为坐标之间的欧几里德距离,费用为海拔之差,现在要求方案使得费用与距离的比值最小,很显然,这个题目 ...
- POJ 2728 Desert King (最优比例生成树)
POJ2728 无向图中对每条边i 有两个权值wi 和vi 求一个生成树使得 (w1+w2+...wn-1)/(v1+v2+...+vn-1)最小. 采用二分答案mid的思想. 将边的权值改为 wi- ...
- Desert King(最优比率生成树)
Desert King Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 22717 Accepted: 6374 Desc ...
- POJ2728 Desert King —— 最优比率生成树 二分法
题目链接:http://poj.org/problem?id=2728 Desert King Time Limit: 3000MS Memory Limit: 65536K Total Subm ...
- 【POJ2728】Desert King 最优比率生成树
题目大意:给定一个 N 个点的无向完全图,边有两个不同性质的边权,求该无向图的一棵最优比例生成树,使得性质为 A 的边权和比性质为 B 的边权和最小. 题解:要求的答案可以看成是 0-1 分数规划问题 ...
随机推荐
- the little schemer 笔记(3)
第三章 cons the magnificent (rember a lat)是什么,其中a是mint,lat是(lamb chops and mint jelly) (lamb chops and ...
- mysql架构解析
架构 mysql是cs架构,在服务端可以启动该mysqld服务,mysqld进程默认监听在tcp:3306.在客户端使用命令行工具mysql或者图形化工具navicat for mysql进行远程连接 ...
- scikit-learning教程(三)使用文本数据
使用文本数据 本指南的目标是探讨scikit-learn 一个实际任务中的一些主要工具:分析二十个不同主题的文本文档(新闻组帖子)集合. 在本节中,我们将看到如何: 加载文件内容和类别 提取适用于机器 ...
- AtCoder Grand Contest 015 C - Nuske vs Phantom Thnook
题目传送门:https://agc015.contest.atcoder.jp/tasks/agc015_c 题目大意: 现有一个\(N×M\)的矩阵\(S\),若\(S_{i,j}=1\),则该处为 ...
- The 17th Zhejiang University Programming Contest Sponsored by TuSimple J
Knuth-Morris-Pratt Algorithm Time Limit: 1 Second Memory Limit: 65536 KB In computer science, t ...
- javascript回调函数那些事~
什么是回调函数? 回调函数就是一个通过函数指针调用的函数.如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数.回调函数不是由该函数的实现方直 ...
- spring boot使用jpa查询mysql数据库的视图时不报错,但查询结果数据总是重复第一条
问题描述: 在数据库里查询到的结果是正常显示的 在程序中返回的结果: 解决方法: 添加行号作为主键: 解决! 我明明是前端啊前端,为啥在搞后台....,总感觉我要在向全栈进发,希望自己有朝一日真的能成 ...
- 在服务端C#如何利用NPOI构建Excel模板
目前本人接触过两种模板导出的方式:(1)C#利用NPOI接口制作Excel模板,在服务端用数据渲染模板(2)在前端利用前人搭建好的框架,利用office编写xml制作模板,在客户端进行数据的渲染,导出 ...
- springdata-jpa 八种查询方法
使用:maven+Spring+jpa+Junit4 查询方式:SQL,JPQL查询,Specification多条件复杂查询 返回类型:list<POJO>,list<Stinrg ...
- Keil简介
最早接触Keil是学习开发8051系列的单片机.Keil C51是Keil公司出品的51系列兼容单片机C语言软件开发系统,与汇编相比,C语言在功能上.结构性.可读性.可维护性上有明显的优势,因而易学易 ...