POJ 1639 Picnic Planning:最小度限制生成树
题目链接:http://poj.org/problem?id=1639
题意:
给你一个无向图,n个节点,m条边,每条边有边权。
让你求一棵最小生成树,同时保证1号节点的度数<=k。
题解:
最小度限制生成树:
(1)不用与1号节点相连的边,跑一次kruskal,得到了deg个连通块。
(2)选取与1相连的deg条边,并使得边尽可能小,将1与这些连通块连起来,得到了一棵deg度最小生成树。
(3)利用当前的deg度最小生成树,求出deg+1度最小生成树。如此重复至k度最小生成树:
I. 在当前生成树上dfs求出:从1出发到i节点路径上的最大边dp[i](除去与1相连的边)。
II. 枚举与1相连且不在生成树内的边,添加一条能使当前生成树变得最小的边。
III. 如果无论如何都无法将生成树变小,则已求出答案,退出循环。
AC Code:
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <algorithm>
#include <map>
#define MAX_N 35
#define MAX_M 905
#define INF 1000000000 using namespace std; struct Edge
{
int s;
int t;
int len;
Edge(int _s,int _t,int _len)
{
s=_s;
t=_t;
len=_len;
}
Edge(){}
friend bool operator < (const Edge &a,const Edge &b)
{
return a.len<b.len;
}
}; int n=,m,k;
int ans=,deg=;
int par[MAX_N];
int lnk[MAX_N];
int minn[MAX_N];
int a[MAX_N][MAX_N];
bool flag[MAX_N][MAX_N];
Edge dp[MAX_N];
Edge edge[MAX_M];
map<string,int> mp; int cal_id(const string &s)
{
if(mp.count(s)>) return mp[s];
mp.insert(pair<string,int>(s,++n));
return n;
} void read()
{
cin>>m;
string s1,s2;
int v;
memset(a,-,sizeof(a));
cal_id("Park");
for(int i=;i<m;i++)
{
cin>>s1>>s2>>v;
int id1=cal_id(s1);
int id2=cal_id(s2);
edge[i]=Edge(id1,id2,v);
if(a[id1][id2]==-) a[id1][id2]=a[id2][id1]=v;
else a[id1][id2]=a[id2][id1]=min(a[id1][id2],v);
}
cin>>k;
} void init_union_find()
{
for(int i=;i<=n;i++)
{
par[i]=i;
}
} int find(int x)
{
return par[x]==x ? x : par[x]=find(par[x]);
} void unite(int x,int y)
{
int px=find(x);
int py=find(y);
if(px==py) return;
par[px]=py;
} bool same(int x,int y)
{
return find(x)==find(y);
} void kruskal()
{
init_union_find();
sort(edge,edge+m);
memset(flag,false,sizeof(flag));
for(int i=;i<m;i++)
{
Edge temp=edge[i];
if(temp.s== || temp.t==) continue;
if(!same(temp.s,temp.t))
{
ans+=temp.len;
unite(temp.s,temp.t);
flag[temp.s][temp.t]=flag[temp.t][temp.s]=true;
}
}
} void cal_mdeg()
{
memset(minn,0x3f,sizeof(minn));
for(int i=;i<=n;i++)
{
if(a[][i]!=-)
{
int p=find(i);
if(a[][i]<minn[p])
{
minn[p]=a[][i];
lnk[p]=i;
}
}
}
for(int i=;i<=n;i++)
{
if(minn[i]<INF)
{
deg++;
ans+=minn[i];
flag[][lnk[i]]=flag[lnk[i]][]=true;
}
}
} void dfs(int x,int p)
{
for(int i=;i<=n;i++)
{
if(flag[x][i] && i!=p)
{
if(dp[i].len==-)
{
if(a[x][i]>dp[x].len) dp[i]=Edge(x,i,a[x][i]);
else dp[i]=dp[x];
}
dfs(i,x);
}
}
} void cal_kdeg()
{
for(int j=deg+;j<=k;j++)
{
dp[]=Edge(-,-,-INF);
for(int i=;i<=n;i++)
{
if(flag[][i]) dp[i]=Edge(-,-,-INF);
else dp[i]=Edge(-,-,-);
}
dfs(,-);
int dst,maxn=-INF;
for(int i=;i<=n;i++)
{
if(a[][i]!=- && dp[i].len-a[][i]>maxn)
{
maxn=dp[i].len-a[][i];
dst=i;
}
}
if(maxn<=) return;
int x=dp[dst].s,y=dp[dst].t;
flag[x][y]=flag[y][x]=false;
flag[][dst]=flag[dst][]=true;
ans-=maxn;
}
} void solve()
{
kruskal();
cal_mdeg();
cal_kdeg();
} void print()
{
cout<<"Total miles driven: "<<ans<<endl;
} int main()
{
read();
solve();
print();
}
POJ 1639 Picnic Planning:最小度限制生成树的更多相关文章
- POJ 1639 Picnic Planning(最小度限制生成树)
Description The Contortion Brothers are a famous set of circus clowns, known worldwide for their inc ...
- POJ 1639 Picnic Planning 最小k度生成树
Picnic Planning Time Limit: 5000MS Memory Limit: 10000K Total Submissions:11615 Accepted: 4172 D ...
- [POJ 1639] Picnic Planning
[题目链接] http://poj.org/problem?id=1639 [算法] 首先,我们可以用深度优先遍历求出1号节点去除后有几个联通块 设共有T个联通块,若T > K则无解,否则 : ...
- poj 1639 Picnic Planning 度限制mst
https://vjudge.net/problem/POJ-1639 题意: 有一群人,他们要去某一个地方,每个车可以装无数个人,给出了n条路,包含的信息有路连接的地方,以及路的长度,路是双向的,但 ...
- poj1639 Picnic Planning 最小度数限制生成树
题意:若干个人开车要去park聚会,可是park能停的车是有限的,为k.所以这些人要通过先开车到其它人家中,停车,然后拼车去聚会.另外,车的容量是无限的,他们家停车位也是无限的. 求开车总行程最短. ...
- poj1639,uva1537,uvalive2099,scu1622,fzu1761 Picnic Planning (最小限制生成树)
Picnic Planning Time Limit: 5000MS Memory Limit: 10000K Total Submissions: 10742 Accepted: 3885 ...
- POJ1639 - Picnic Planning
原题链接 Description 给出一张个点的无向边权图并钦定点,求使得点的度不超过的最小生成树. Solution 首先无视掉与相连的所有边,原图会变成若干互不连通的个块.对每个块分别求MST,再 ...
- K度限制MST poj 1639
/* k度限制MST:有一个点的度<=k的MST poj 1639 要求1号点的度不超过k 求MST 我们先把1号点扔掉 跑MST 假设有sum个连通分支 然后把这sum个分支连到1上 就得到了 ...
- luogu P5633 最小度限制生成树 wqs二分
LINK:最小度限制生成树 还是WQS二分的模板题 不过相当于我WQS二分的复习题. 对于求出强制k个的答案 dp能做不过复杂度太高了. 世界上定义F(x)表示选出x个的答案 画成图像 其实形成了一个 ...
随机推荐
- Linux下性能分析工具汇总
来自:http://os.51cto.com/art/201104/253114.htm 本文讲述的是:CPU性能分析工具.Memory性能分析工具.I/O性能分析工具.Network性能分析工具. ...
- MySQL_常用SQL语句
1.按小时统计的语句 select concat(date_format(gmt_create, "%Y-%m-%d %k:00~"), hour(gmt_create)+1, & ...
- Android linux kernel privilege escalation vulnerability and exploit (CVE-2014-4322)
In this blog post we'll go over a Linux kernel privilege escalation vulnerability I discovered which ...
- 转载了个js代码
document.selection.createRange方法 document.selection.createRange() 根据当前文字选择返回 TextRange 对象,或根据控件选择返回 ...
- Linux下文件的堵塞与非堵塞对部分系统调用的影响
1.基本概念 所谓的堵塞,即内核在对文件操作I/O系统调用时.假设条件不满足(可能须要产生I/O),则内核会将该进程挂起.非堵塞则是发现条件不满足就会马上返回. 此外须要注意的是非堵塞并非轮询.不然就 ...
- Linux基础知识之挂载详解(mount,umount及开机自动挂载)
Linux基础知识之挂载详解(mount,umount及开机自动挂载) 转载自:http://www.linuxidc.com/Linux/2016-08/134666.htm 挂载概念简述: 根文件 ...
- Vue.js 2 入门与提高(一)
** Vue.js -- 渐进式前端框架 ** Vue.js作为一个后起的前端框架,借鉴了Angular .React等现代前端框架/库的诸多特点,并且 取得了相当不错的成绩. Vue.js的定位是一 ...
- 第三篇:Logstash 安装配置
Logstash 简介: Logstash 是一个实时数据收集引擎,可收集各类型数据并对其进行分析,过滤和归纳.按照自己条件分析过滤出符合数据导入到可视化界面.Logstash 建议使用java1.8 ...
- 05 linux下编译Memcached
一: linux下编译Memcached(1)先在linux上安装gcc,cmake,autoconf,libtool工具,以后会使用到. 安装命令为:yum install gcc make aut ...
- mysql主从实验
实验一: 服务器A与服务器B是主从关系,关联的库有test1,.test2库,主从一直运行正常,此时由于业务的需求,需再增加test3库,并让其仍然维持主从关系,应该怎么做? 步骤一: 修改从库服务器 ...