原题链接

Description

给出一张n(n≤20)个点的无向边权图并钦定点P,求使得点P的度不超过k的最小生成树。

Solution

首先无视掉与P相连的所有边,原图会变成若干互不连通的t个块。对每个块分别求MST,再从每个块向P连一条最小的边,这样就得到了一个t度最小生成树。不存在度数比t小的生成树了,因为这t个部分只能通过P来连通。这个比较容易理解。

然后考虑如何从t度MST转移为t+1度MST。我们可以尝试加入一条与P相连的边,图中会出现一个环,从环上再删掉一条最大的边就可以得到一棵t+1度生成树。枚举所有与P相连且不在当前的生成树上的边,最小化总权值就好啦。每次转移可以通过BFS找出路径(P,v)上的最大边是哪条。

时间复杂度O(mlogm+kn)。

Code

//Picnic Planning
#include <cstdio>
#include <iostream>
#include <cstring>
#include <algorithm>
#include <map>
using namespace std;
map<string,int> p;
int const N=30;
int const INF=0x3F3F3F3F;
int n,m,rt;
//图的存储
int e[N][N],cnt;
struct edge{int u,v,c; edge(int u1=0,int v1=0,int c1=0){u=u1,v=v1,c=c1;};} ed[N*N],mx[N];
void edAdd(int u,int v,int c)
{
e[u][v]=e[v][u]=min(e[u][v],c);
if(u!=rt&&v!=rt) ed[++cnt]=edge(u,v,c);
}
//划分块相关
int clCnt,f[N];
void extend(int u)
{
for(int v=1;v<=n;v++)
if(e[u][v]<INF&&!f[v]) f[v]=f[u],extend(v);
}
//生成树相关
int sum; bool tr[N][N];
void trSet(int u,int v,bool opt) {tr[u][v]=tr[v][u]=opt; sum+=(opt?1:-1)*e[u][v];}
bool cmpC(edge x,edge y) {return x.c<y.c;}
int fa[N];
int find(int u) {return fa[u]==u?u:fa[u]=find(fa[u]);}
void Kruskal()
{
sort(ed+1,ed+cnt+1,cmpC);
for(int u=1;u<=n;u++) fa[u]=u;
for(int i=1;i<=cnt;i++)
{
int u=ed[i].u,v=ed[i].v;
if(find(u)==find(v)) continue;
fa[find(v)]=find(u); trSet(u,v,true);
}
}
//生成树转移相关
int toRt[N];
int q[N],op,cl;
void bfs()
{
op=cl=0; memset(mx,0,sizeof mx);
q[++cl]=rt; mx[rt].c=-1;
while(op<cl)
{
int u=q[++op];
for(int v=1;v<=n;v++)
{
if(!tr[u][v]||mx[v].c) continue;
q[++cl]=v; mx[v]=e[u][v]>mx[u].c?edge(u,v,e[u][v]):mx[u];
}
}
}
int main()
{
scanf("%d",&m);
n=0; rt=0; memset(e,0x3F,sizeof e);
for(int i=1;i<=m;i++)
{
string s1,s2; int c;
cin>>s1; cin>>s2; scanf("%d",&c);
if(!p[s1]) p[s1]=++n; if(!p[s2]) p[s2]=++n;
if(s1=="Park") rt=p[s1]; if(s2=="Park") rt=p[s2];
edAdd(p[s1],p[s2],c);
}
int k; scanf("%d",&k);
//划分块
f[rt]=n+1; clCnt=0;
for(int i=1;i<=n;i++) if(f[i]==0) f[i]=++clCnt,extend(i);
//建立clCnt度MST
sum=0; Kruskal();
for(int v=1;v<=n;v++) if(e[rt][v]<e[rt][toRt[f[v]]]) toRt[f[v]]=v;
for(int i=1;i<=clCnt;i++) trSet(rt,toRt[i],true);
//由t-1度MST转化为t度MST
for(int t=clCnt+1;t<=k;t++)
{
int v0=0,dlt=INF; bfs();
for(int v=1;v<=n;v++)
if(!tr[rt][v]&&e[rt][v]<INF)
if(e[rt][v]-mx[v].c<dlt) dlt=e[rt][v]-mx[v].c,v0=v;
if(dlt>=0) break;
trSet(mx[v0].u,mx[v0].v,false);
trSet(rt,v0,true);
}
printf("Total miles driven: %d",sum);
return 0;
}

P.S.

又是坑爹的读入…还要注意细节

POJ1639 - Picnic Planning的更多相关文章

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

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

  2. poj1639 Picnic Planning,K度限制生成树

    题意: 矮人虽小却喜欢乘坐巨大的轿车,车大到能够装下不管多少矮人.某天,N(N≤20)个矮人打算到野外聚餐.为了集中到聚餐地点,矮人A 要么开车到矮人B 家中,留下自己的轿车在矮人B 家,然后乘坐B ...

  3. POJ-1639 Picnic Planning 度数限制最小生成树

    解法参考的论文:https://wenku.baidu.com/view/8abefb175f0e7cd1842536aa.html 觉得网上的代码好像都是用邻接矩阵来实现的,觉得可能数据量大了会比较 ...

  4. POJ1639 Picnic Planning (限制入度最小生成树)

    节点1是有度数限制的,把节点1去掉,就会形成若干个连通块,在每个连通块内部求最小生成树(prim算法实现),并求出每个连通块与1相连的最短的边,这样形成了初始状态的生成树. 假设(1,x)这条边没在生 ...

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

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

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

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

  7. 【POJ 1639】 Picnic Planning (最小k度限制生成树)

    [题意] 有n个巨人要去Park聚会.巨人A和先到巨人B那里去,然后和巨人B一起去Park.B君是个土豪,他家的停车场很大,可以停很多车,但是Park的停车场是比较小.只能停k辆车.现在问你在这个限制 ...

  8. poj 1639 Picnic Planning 度限制mst

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

  9. Picnic Planning POJ - 1639(最小k度生成树)

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

随机推荐

  1. 重温MFC

    1. Button控件 2. 旋转和高级编辑控件 3. 标签控件和属性页 4. 列表控件 5. 树控件 6. 进度条控件和滑动条控件 7. 滚动条 8.工具栏和状态栏

  2. Linux实践篇--crontab定时任务

    原文出处:http://www.cnblogs.com/tracy/archive/2011/12/27/2303788.html.感谢作者的无私分享 一.  Crontab 介绍           ...

  3. Linux指令--ifconfig

    许多windows非常熟悉ipconfig命令行工具,它被用来获取网络接口配置信息并对此进行修改.Linux系统拥有一个类似的工具,也就是ifconfig(interfaces config).通常需 ...

  4. Newtonsoft.Json 版本冲突时解决方案

    如果同一项目中不同第三方类库分别使用了不同版本的Newtonsoft.Json的情况下,可以在主项目配置文件中添加以下节点,将0.0.0.0-11.0.0.0此区间的Newtonsoft.Json使用 ...

  5. JAVA中创建线程的三种方法及比较

    JAVA中创建线程的方式有三种,各有优缺点,具体如下: 一.继承Thread类来创建线程 1.创建一个任务类,继承Thread线程类,因为Thread类已经实现了Runnable接口,然后重写run( ...

  6. PHP获取字符串编码与转码

    (⊙o⊙)-编辑器保存的是utf8的文档取出来的字符串是gbk编码?问题很多,字符串转换为utf8的话在浏览器输出乱码 $e=mb_detect_encoding($d,array('GB2312', ...

  7. curl访问nagios中Host Status Details For All Host Groups页面的方法

    由于进入nagios要输入用户名与密码才能进入,故用curl模拟输入用户名与密码取得当前所有主机的报警信息: # curl -u nagiosadmin:password http://192.168 ...

  8. Linux命令:useradd

    Linux下:useradd 等价于 adduser     Aix下:useradd 来自为知笔记(Wiz)

  9. MVC登出友情提示

    好奇心害死猫,话说12月30号,阳历17年末,行至小镇农户,有羊在鸣,由于好奇心泛滥,循声而至,发现一只被拴住的羊,好心喂草,不料反被舔了一口,得亏抽手及时,不然小拇指估计被咬掉,没错,是咬掉,即使如 ...

  10. [C#] C# 与 MongoDB 的 CURD

    C# 与 MongoDB 的 CURD static void Main(string[] args) { //建立连接 var client = new MongoClient(); //建立数据库 ...