树形DP 树的最小支配集,最小点覆盖与最大独立集
最小支配集:
从V中选取尽量少的点组成一个集合,让V中剩余的点都与取出来的点有边相连。
(点)
最小点覆盖:
从V中选取尽量少的点组成一个集合V1,让所有边(u,v)中要么u属于V1,要么v属于V1
(边)
最大独立集:
从V中选取尽量多的点组成一个集合,让这些点中间没有边项链,也就是说对于任何一条边,u,v不能同时属于集合V1.
1.贪心算法
首先选取一个点为根节点,求出所有节点对应的DFS序列,按照所得序列反向进行贪心,这样保证对于每个点来说,当子树都被处理过之后才会处理该节点
int p[MAXN];
bool select[MAXN];
int newpos[MAXN];
int now, n, m;
//最小支配集
int greedy1()
{
bool s[MAXN] = { };
bool set[MAXN] = { };
int ans = ;
int i;
for (i = n - ; i >= ; i--)
{
int t = newpos[p[t]];
if (!s[t])
{
if (!set[p[t]])
{
set[p[t]] = true;
ans++;
}
s[t] = s[p[t]] = s[p[p[t]]] = true;
}
}
return ans;
}
//最小点覆盖
int greedy2()
{
bool s[MAXN] = { };
bool set[MAXN] = { };
int ans = ;
for (int i = n - ; i >= ; i--)//不可以检查根节点,p[root] = root
{
int t = newpos[i];
if (!s[t] && !s[p[t]])
{
set[p[t]] = true;
ans++;
s[t] = s[p[t]] = true;
}
}
}
//最大独立集
int greedy3()
{
int ans = ;
bool s[MAXN] = { };
bool set[MAXN] = { };
for (int i = ; i >= ; i--)
{
int t = newpos[i];
if (!s[t])
{
set[t] = true;
ans++;
s[t] = s[p[t]] = true;
}
}
return ans;
}
2.树形DP
1.dp[i][0]: 表示点i属于支配集,并且以点i为根的子树都被覆盖了的情况下支配集中包含点最少的个数
2.dp[i][1] i不属于支配集,而且以i为根的子树都被覆盖而且i被其中不少于一个子节点覆盖情况下支配集所包含最少点的个数
3.dp[i][2] i不属于支配集,而且i为根的子树都被覆盖而且I没有被子节点覆盖的情况下支配集报验最少点的个数
对于第一情况,对子节点无限制
dp[i][0] = 1 + 西格玛min(dp[u][0],dp[u][1],dp[u][2]) p[u] = t
对于第二红情况, 如果i没有子节点那么dp[i][1] = INF,子节点必须被覆盖,所以和状态dp[i][2]无关的!
dp[i][1] = 西格玛min(dp[u][0],dp[u][1]) + inc
如果选取了某一个dp[u][0] inc = 0;
else inc = min(dp[u][0] - dp[u][1])
选取的时候注意如果全部选的都是dp[u][1]那父节点没办法被覆盖啦!所以判断一下
对于第三种情况,i不属于支配集,i的子树都被覆盖,说明i和i的儿子都不是支配集的!
dp[i][2] = 西格玛dp[u][1]
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<sstream>
#include<algorithm>
#include<queue>
#include<deque>
#include<iomanip>
#include<vector>
#include<cmath>
#include<map>
#include<stack>
#include<set>
#include<functional>
#include<fstream>
#include<memory>
#include<list>
#include<string>
using namespace std;
typedef long long LL;
typedef unsigned long long ULL; #define N 31
#define INF 1000000009
#define eps 0.00000001
#define sf(a) scanf("%d",&a)
const int MAXN = 1e5 + ; int dp[MAXN][];
int head[MAXN];
void DP(int u, int p)
{
dp[u][] = ;
dp[u][] = ;
bool s = false;
int sum = , inc = INF;
int k;
for (k = head[u]; k != -; k = E[k].next)
{
int to = E[k].to;
if (to == p)
continue;
DP(to, u);
dp[u][] += min(dp[to][], dp[to][], dp[to][]);
if (dp[to][] < dp[to][])
{
s = true;
sum += dp[to][];
}
else
{
sum += dp[to][];
inc = min(inc, dp[to][] - dp[to][]);
}
if (dp[to][] != INF&&dp[u][] != INF)
dp[u][] += dp[to][];
else
dp[u][] = INF;
}
if (inc == INF && !s)
dp[u][] = INF;
else
{
dp[u][] = sum;
if (!s) dp[u][] += inc;
}
}
对于最小点覆盖
dp[u][0] u点被覆盖
dp[u][1] u点没被覆盖
int dp[MAXN][];
int head[MAXN];
void DP(int u, int p)
{
dp[u][] = ;
dp[u][] = ;
int k, to;
for (k = head[u]; k != -; k = E[k].next)
{
to = E[k].to;
if (to == p)
continue;
DP(to, u);
dp[u][] += min(dp[to][], dp[to][]);
dp[u][] += dp[to][];
}
}
树形DP 树的最小支配集,最小点覆盖与最大独立集的更多相关文章
- 树形DP求树的最小支配集,最小点覆盖,最大独立集
一:最小支配集 考虑最小支配集,每个点有两种状态,即属于支配集合或者不属于支配集合,其中不属于支配集合时此点还需要被覆盖,被覆盖也有两种状态,即被子节点覆盖或者被父节点覆盖.总结起来就是三种状态,现对 ...
- 求树的最大独立集,最小点覆盖,最小支配集 贪心and树形dp
目录 求树的最大独立集,最小点覆盖,最小支配集 三个定义 贪心解法 树形DP解法 (有任何问题欢迎留言或私聊&&欢迎交流讨论哦 求树的最大独立集,最小点覆盖,最小支配集 三个定义 最大 ...
- POJ3659 Cell Phone Network(树上最小支配集:树型DP)
题目求一棵树的最小支配数. 支配集,即把图的点分成两个集合,所有非支配集内的点都和支配集内的某一点相邻. 听说即使是二分图,最小支配集的求解也是还没多项式算法的.而树上求最小支配集树型DP就OK了. ...
- 树形dp(最小支配集)
http://poj.org/problem?id=3659 #include<iostream> #include<cstring> #include<algorith ...
- POJ 3659 Cell Phone Network / HUST 1036 Cell Phone Network(最小支配集,树型动态规划,贪心)-动态规划做法
POJ 3659 Cell Phone Network / HUST 1036 Cell Phone Network(最小支配集,树型动态规划,贪心) Description Farmer John ...
- POJ 3398 Perfect Service(树型动态规划,最小支配集)
POJ 3398 Perfect Service(树型动态规划,最小支配集) Description A network is composed of N computers connected by ...
- POJ 3659 Cell Phone Network(树的最小支配集)(贪心)
Cell Phone Network Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 6781 Accepted: 242 ...
- 树的问题小结(最小生成树、次小生成树、最小树形图、LCA、最小支配集、最小点覆盖、最大独立集)
树的定义:连通无回路的无向图是一棵树. 有关树的问题: 1.最小生成树. 2.次小生成树. 3.有向图的最小树形图. 4.LCA(树上两点的最近公共祖先). 5.树的最小支配集.最小点覆盖.最大独立集 ...
- POJ 3398 Perfect Service --最小支配集
题目链接:http://poj.org/problem?id=3398 这题可以用两种上述讲的两种算法解:http://www.cnblogs.com/whatbeg/p/3776612.html 第 ...
随机推荐
- c#.net 正则匹配以特定字符串开头,以特定字符串结尾
string[] unit = Getunit(result40, "(?<=(开始字符串))[.\\s\\S]*?(?=(结束字符串))"); private string ...
- 微软最新的Web服务器Katana发布了版本3
Katana 项目入门 Howard Dierking 当 ASP.NET 首次在 2002 年发布时,时代有所不同. 那时,Internet 仍处于起步阶段,大约有 5.69 亿用户,每个用户平均每 ...
- json三层解析(数组解析)
里面多了数组,所以用到了JOSNArray package com.xykj.weather; import java.io.BufferedReader; import java.io.IOExce ...
- Uml 建模 一(类图建模和startuml的使用)
本文将分三个部分介绍Uml建模:Uml建模的作用.类图.startuml的使用 Uml的作用 本文以java为例介绍Uml,在当前的软件开发中大多数使用面向对象开发(OO),面向对象的就是将现实世界中 ...
- VS2017 移动开发(Android and IOS) 序
序 公司原因,要求用C#开发移动端app,老板觉得用现在会的C#做会比较快... 从零开始,折腾一个多星期,重装系统三遍(强迫症),其它各种折腾,终于手机运行上了第一个APP,看看就好... 不得不吐 ...
- java将字段映射成另一个字段,关于 接口传参 字段不对应转换
在接口开发中我们经常会遇到一个问题,打个比方,我们的实体类A中有两个字段user和pwd但是接口中需要username和password这怎么办呢,我想到了两种方法:1.新创建一个实体类B或者new一 ...
- js 动态添加表单 table tr
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- 一篇文章告你python能做什么,该不该学?好不好学?适不适合学?
一.python好学吗?简单吗?容易学吗?没有编程的领取能学吗? 最近有很多小伙伴都在问我这些问题.在这里,我想说,python非常简单易学. 1,简单, Python 非常易于读写,开发者可以把更多 ...
- No-3.算数运算符
01. 算数运算符 算数运算符是 运算符的一种 是完成基本的算术运算使用的符号,用来处理四则运算 运算符 描述 实例 + 加 10 + 20 = 30 - 减 10 - 20 = -10 * 乘 1 ...
- 为了安全请不要随意在页面中设置validateRequest="false"
为了安全请不要随意在页面中设置validateRequest="false" 分类: ASP.NET2009-04-12 17:24 531人阅读 评论(0) 收藏 举报 asp. ...