题目链接: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:最小度限制生成树的更多相关文章

  1. POJ 1639 Picnic Planning(最小度限制生成树)

    Description The Contortion Brothers are a famous set of circus clowns, known worldwide for their inc ...

  2. POJ 1639 Picnic Planning 最小k度生成树

    Picnic Planning Time Limit: 5000MS   Memory Limit: 10000K Total Submissions:11615   Accepted: 4172 D ...

  3. [POJ 1639] Picnic Planning

    [题目链接] http://poj.org/problem?id=1639 [算法] 首先,我们可以用深度优先遍历求出1号节点去除后有几个联通块 设共有T个联通块,若T > K则无解,否则 : ...

  4. poj 1639 Picnic Planning 度限制mst

    https://vjudge.net/problem/POJ-1639 题意: 有一群人,他们要去某一个地方,每个车可以装无数个人,给出了n条路,包含的信息有路连接的地方,以及路的长度,路是双向的,但 ...

  5. poj1639 Picnic Planning 最小度数限制生成树

    题意:若干个人开车要去park聚会,可是park能停的车是有限的,为k.所以这些人要通过先开车到其它人家中,停车,然后拼车去聚会.另外,车的容量是无限的,他们家停车位也是无限的. 求开车总行程最短. ...

  6. poj1639,uva1537,uvalive2099,scu1622,fzu1761 Picnic Planning (最小限制生成树)

    Picnic Planning Time Limit: 5000MS   Memory Limit: 10000K Total Submissions: 10742   Accepted: 3885 ...

  7. POJ1639 - Picnic Planning

    原题链接 Description 给出一张个点的无向边权图并钦定点,求使得点的度不超过的最小生成树. Solution 首先无视掉与相连的所有边,原图会变成若干互不连通的个块.对每个块分别求MST,再 ...

  8. K度限制MST poj 1639

    /* k度限制MST:有一个点的度<=k的MST poj 1639 要求1号点的度不超过k 求MST 我们先把1号点扔掉 跑MST 假设有sum个连通分支 然后把这sum个分支连到1上 就得到了 ...

  9. luogu P5633 最小度限制生成树 wqs二分

    LINK:最小度限制生成树 还是WQS二分的模板题 不过相当于我WQS二分的复习题. 对于求出强制k个的答案 dp能做不过复杂度太高了. 世界上定义F(x)表示选出x个的答案 画成图像 其实形成了一个 ...

随机推荐

  1. vim与windows/linux之间的复制粘贴小结

    vim与windows/linux之间的复制粘贴小结 用 vim这么久了,始终也不知道怎么在vim中使用系统粘贴板,通常要在网上复制一段代码都是先gedit打开文件,中键粘贴后关闭,然后再用vim打开 ...

  2. Sublime Text3 运行python(转)

    From:http://blog.csdn.net/hun__ter/article/details/51223031 安装sublime text3后,按Ctrl+b无法运行python文件. 解决 ...

  3. Boost学习总结(一)VS2010环境下编译STLport和Boost

    Boost学习总结(一)VS2010环境下编译STLport和Boost Boost简介 Boost库是一个功能强大.构造精巧.跨平台.开源并且完全免费的C++程序库.1998年,Beman G.Da ...

  4. win10下VS2010中文输入法切换为英文卡死

    中文输入法下输入英文,VS2010会出现卡顿现象,之后会出现一大串的重复英文字母. win10下VS2010安装的助手VAssit系统不兼容,而win7下不会出现上述问题. 解决办法:卸载安装的助手V ...

  5. NumPy入门基础【2】

    通用函数ufunc 一元ufunc举例: 1.abs.fabs:计算绝对值,fabs更快 2.sqrt:计算各元素的平方根,相当于arr0.5 3.square:计算各元素的平方根,相当远arr2 4 ...

  6. vue-cli (vue脚手架)

    vue-cli(脚手架):它可以自动生成目录 1.在网速不佳的情况下可以安装cnpm(淘宝镜像)如果网速快可以不用安装cnpm直接进行下一步操作 第一步:在命令行执行(全局安装cnpm) npm in ...

  7. 摘要: CentOS 6.5搭建Redis3.2.8伪分布式集群

    from https://my.oschina.net/ososchina/blog/856678     摘要: CentOS 6.5搭建Redis3.2.8伪分布式集群 前言 最近在服务器上搭建了 ...

  8. Netty实现java多线程Post请求解析(Map参数类型)—SKY

    netty解析Post的键值对 解析时必须加上一个方法,ch.pipeline().addLast(new HttpObjectAggregator(2048)); 放在自己的Handel前面. ht ...

  9. 九度OJ 1191:矩阵最大值 (矩阵计算)

    时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:2361 解决:1179 题目描述: 编写一个程序输入一个mXn的矩阵存储并输出,并且求出每行的最大值和每行的总和. 要求把每行总和放入每行最 ...

  10. iOS9 3D Touch使用

    http://www.cnblogs.com/zhanglinfeng/p/5133939.html