原题链接

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. mybatis自动生成java代码

    SSM框架没有DB+Record模式,写起来特别费劲,只能用下面的方法勉强凑合. 上图中,*.jar为下载的,src为新建的空白目录,.xml配置如下. <?xml version=" ...

  2. c:if true、false都显示

    看了半天,最后发现jstl标签库没有引入! <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core ...

  3. Can’t open /dev/* exclusively. Mounted filesystem?解决

    1 错误提示:Can’t open /dev/* exclusively. Mounted filesystem? 做完软件RAID之后,根据鸟哥书上的操作,其实没有真正删除软件RAID,导致出现如下 ...

  4. Web应用基础

    B-S架构 架构的发展 1,c/s架构 (client客户端-server服务端)         (胖客户端:要求客户端运行业务:把业务放到服务器端,则是瘦客户端)         典型的c/s应用 ...

  5. 【转】对GAMIT/GLOBK的基本认识

    1.1   GAMIT/GLOBK软件可从网络上申请下载.该软件功能强大,用途广泛,一般包括精确定位,大气层可降水汽估计和空间电离层变化分析等.后两种用途只需要用到GAMIT模块,精确定位则还需要GL ...

  6. JDK配置测试

    JDK配置测试 介绍两种JDK配置方式: 一:大多数人配置方法 1.下载JDKhttps://www.baidu.com2.配置环境变量单击"计算机-属性-高级系统设置",单击&q ...

  7. 数据库备份shell脚本

    法一: #!/bin/bash [ ! -d /server/backup ] && mkdir /server/backup mysqldump -u root -A -B > ...

  8. Codeforce A. Fair Game

    A. Fair Game time limit per test 1 second memory limit per test 256 megabytes input standard input o ...

  9. 《.NET 设计规范》第 8 章:使用规范

    第 8 章:使用规范 8.1 数组 要在公共 API 中优先使用集合,避免使用数组. 不要使用只读的数组字段.虽然字段本身是只读的,用户不能修改它们,但用户可以修改数组中的元素. 考虑使用不规则数组, ...

  10. 单元测试系列:如何使用JUnit+JaCoCo+EclEmma完成单元测试

    更多原创测试技术文章同步更新到微信公众号 :三国测,敬请扫码关注个人的微信号,感谢!   原文链接:http://www.cnblogs.com/zishi/p/6726664.html -----如 ...