Description

  

  传送门

  

  

  

  

  

Solution

  

  题目要求相同颜色的点必须在一个连通块中,但会有多个颜色同属一个连通块使得解更优的情况。

  

  想一想DP能否行得通:设\(g_i\)表示已考虑颜色状态为\(i\)时,最小合法方案的代价。

  

  首先,\(g_i\)可以有一个直观的初值:由颜色属于\(i\)的点构建的一棵最小生成树的边权和。(初始化)

  

​  接下来,如何考虑两部分颜色各自的连通块合起来作为最优解的情况?(两子集合并更新)

  

​  更新\(g_i\)时,我们枚举\(i\)的两个不相交子集\(s_1,s_2\)且\(s_1|s_2=i\),并用\(g_{s_1}+g_{s_2}\)来更新\(g_i\),即直接尝试将两个连通块取交。不需要担心两部分会有重复计算某一条边的情况,因为这是子集枚举DP,有边重复计算的转移自然不会成为最优转移,最优的转移一定会被枚举到(或者就是初值最优)。

  

​  以上两种方法并用,就可以顺利DP出\(g\)。

  

​  答案就是\(g_{all}\)。\(all\)是包含所有颜色的集合。

  

​  对于\(g\)的初值,直接用斯坦纳树计算所有关键点的不同组合的最小生成树即可。

  

  

  

Code

  

#include <cstdio>
#include <algorithm>
#include <queue>
using namespace std;
const int N=1005,M=3005,C=10,INF=1e9;
int n,m,p,col[N],tin[N][2],colst[C+1];
int idcnt,id[N],rep[C+1];
int clis[C+1],ccnt;
int f[N][1<<C],g[1<<C];
queue<int> q;
bool inq[N];
int h[N],tot;
struct Edge{int v,w,next;}e[M*2];
inline int bit(int i){return !i?0:(1<<(i-1));}
inline bool in(int st,int i){return (st>>(i-1))&1;}
inline void addEdge(int u,int v,int w){
e[++tot]=(Edge){v,w,h[u]}; h[u]=tot;
e[++tot]=(Edge){u,w,h[v]}; h[v]=tot;
}
void read(){
scanf("%d%d%d",&n,&m,&p);
for(int i=1,u,v,w;i<=m;i++){
scanf("%d%d%d",&u,&v,&w);
addEdge(u,v,w);
}
for(int i=1,c,u;i<=p;i++){
scanf("%d%d",&c,&u);// c ?
tin[i][0]=c; tin[i][1]=u;
clis[++ccnt]=c;
}
sort(clis+1,clis+1+ccnt);
ccnt=unique(clis+1,clis+1+ccnt)-clis-1;
for(int i=1,u,c;i<=p;i++){
c=lower_bound(clis+1,clis+1+ccnt,tin[i][0])-clis;
u=tin[i][1];
col[u]=c;
id[u]=++idcnt;
rep[c]=u;
colst[c]|=bit(id[u]);
}
}
void spfa(int st){
while(!q.empty()){
int u=q.front(); q.pop();
inq[u]=false;
for(int i=h[u],v;i;i=e[i].next){
v=e[i].v;
int ns=st|bit(id[v]);
if(f[u][st]+e[i].w<f[v][ns]){
f[v][ns]=f[u][st]+e[i].w;
if(!inq[v]){
inq[v]=true;
q.push(v);
}
}
}
}
}
void steinerTree(){
int all=1<<p;
for(int i=1;i<=n;i++){
for(int j=0;j<all;j++) f[i][j]=INF;
f[i][bit(id[i])]=0;
}
for(int j=1;j<all;j++){
for(int i=1;i<=n;i++)
if(!(id[i]&&!in(j,id[i]))){
for(int sub=(j-1)&j;sub;sub=(sub-1)&j){
int x=sub|bit(id[i]),y=(j^sub)|bit(id[i]);
if(f[i][x]+f[i][y]<f[i][j])
f[i][j]=f[i][x]+f[i][y];
}
if(f[i][j]!=INF)
q.push(i),inq[i]=true;
}
spfa(j);
}
}
void solve(){
int all=1<<ccnt;
for(int i=1;i<all;i++){
int st=0,rt=0;
for(int j=1;j<=ccnt;j++)
if(in(i,j))
st|=colst[j],rt=rep[j];
g[i]=f[rt][st];
for(int sub=(i-1)&i;sub;sub=(sub-1)&i)
g[i]=min(g[i],g[sub]+g[i^sub]);
}
printf("%d\n",g[all-1]);
}
int main(){
read();
steinerTree();
solve();
return 0;
}

【BZOJ4006】【JLOI2015】管道连接的更多相关文章

  1. BZOJ4006 JLOI2015 管道连接(斯坦纳树生成森林)

    4006: [JLOI2015]管道连接 Time Limit: 30 Sec Memory Limit: 128 MB Description 小铭铭最近进入了某情报部门,该部门正在被如何建立安全的 ...

  2. [BZOJ4006][JLOI2015]管道连接 状压dp+斯坦纳树

    4006: [JLOI2015]管道连接 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1020  Solved: 552[Submit][Statu ...

  3. [bzoj4006][JLOI2015]管道连接_斯坦纳树_状压dp

    管道连接 bzoj-4006 JLOI-2015 题目大意:给定一张$n$个节点$m$条边的带边权无向图.并且给定$p$个重要节点,每个重要节点都有一个颜色.求一个边权和最小的边集使得颜色相同的重要节 ...

  4. BZOJ4006: [JLOI2015]管道连接(斯坦纳树,状压DP)

    Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1171  Solved: 639[Submit][Status][Discuss] Descripti ...

  5. BZOJ4006 [JLOI2015]管道连接

    裸的状压DP 令$f_S$表示包含颜色集合S的最小斯坦纳生成森林的值,于是有: $$f_S=\min\{f_S,f_s+f_{S-s}|s\subset S\}$$ 然后嘛...还是裸的斯坦纳树搞搞. ...

  6. BZOJ_4006_[JLOI2015]管道连接_斯坦纳树

    BZOJ_4006_[JLOI2015]管道连接_斯坦纳树 题意: 小铭铭最近进入了某情报部门,该部门正在被如何建立安全的通道连接困扰. 该部门有 n 个情报站,用 1 到 n 的整数编号.给出 m ...

  7. luogu P3264 [JLOI2015]管道连接

    LINK:管道连接 一张无向图 有P个关键点 其中有K个集合 各个集合要在图中形成联通块 边有边权 求最小代价. 其实还是生成树问题 某个点要和某个点要在生成树中 类似这个意思. 可以发现 是斯坦纳树 ...

  8. 【bzoj4006】[JLOI2015]管道连接 斯坦纳树+状压dp

    题目描述 给出一张 $n$ 个点 $m$ 条边的无向图和 $p$ 个特殊点,每个特殊点有一个颜色.要求选出若干条边,使得颜色相同的特殊点在同一个连通块内.输出最小边权和. 输入 第一行包含三个整数 n ...

  9. 【bzoj4006】[JLOI2015]管道连接(斯坦纳树+dp)

    题目链接 题意: 给出\(n\)个点,\(m\)条边,同时给出\(p\)个重要的点以及对应特征. 现在要选出一些边,问使得这\(p\)个所有特征相同的点相连,问最小代价. 思路: 斯坦纳树的应用场景一 ...

  10. [JLOI2015]管道连接

    题目描述 小铭铭最近进入了某情报部门,该部门正在被如何建立安全的通道连接困扰.该部门有 n 个情报站,用 1 到 n 的整数编号.给出 m 对情报站 ui;vi 和费用 wi,表示情报站 ui 和 v ...

随机推荐

  1. 2015第六届蓝桥杯C/C++ B组

    奖券数目:枚举 有些人很迷信数字,比如带“4”的数字,认为和“死”谐音,就觉得不吉利.虽然这些说法纯属无稽之谈,但有时还要迎合大众的需求.某抽奖活动的奖券号码是5位数(10000-99999),要求其 ...

  2. mpstat命令详解

    基础命令学习目录首页 原文链接:https://www.cnblogs.com/ggjucheng/archive/2013/01/13/2858775.html 简介 mpstat是Multipro ...

  3. Python20-Day05

    一.模块与包 1.模块 什么是模块? 在python中,模块可以分为四个通用类别: 1. 使用python编写的.py文件 2. 已经被编译为共享库或DLL的c或者c++扩展 3. 把一系列模块组织到 ...

  4. pycharm 打开两个项目

    1.之前用打开文件,选择一直选不到整个项目,打开也都是在已经打开的项目窗口中加载一个文件. 2.网上有介绍说设置,但是自己的pycharm经过中文汉化,许多设置项已消失不见... 3.最近打开,直接操 ...

  5. Python写一个根据日期计算是星期几的模块

    import datetimedef get_week_day(date): week_day = { 0: '星期一', 1: '星期二', 2: '星期三', 3: '星期四', 4: '星期五' ...

  6. 《Spring2之站立会议9》

    <Spring2之站立会议9> 昨天,添加了注册界面: 今天,添加了表情库: 遇到的问题:由于资源有限,感觉设计的不完美并且途中也遇到了好多问题.

  7. 《[C#] int与System.Int32有什么区别》

    最近园里的TeamOne写了一篇<[C#] int与System.Int32有什么区别>,发现里面有不少精彩的评论,所以忍不住想这篇文章总结一下:> 本文的主要参考资料: 1.< ...

  8. self和super关键字介绍

    1.self和super OC提供两个保留字self 和 super ,用在方法定义中 OC语言中的self, 就相当于C++和Java中的this指针,学会使用self 首先要搞清楚属性这一概念以及 ...

  9. JS计算两个日期之间的天数,时间差计算

    1.日期之间的天数计算 //计算天数差的函数,通用 function DateDiff(sDate1, sDate2) { //sDate1和sDate2是2017-9-25格式 var aDate, ...

  10. Sprint2的每日更新

    按照以下过程进行 ProductBacklog:继续向下细化 Sprint 计划会议:确定此次冲刺要完成的目标 Sprint Backlog:新的冲刺要完成的内容 任务认领 Sprint周期 看板:一 ...