Ex 6_21 最小点覆盖问题_第八次作业
子问题定义: 对于图中的每个结点,有两种状态,即属于最小点覆盖和不属于最小点覆盖,定义minSet[i][0]表示结点i属于点覆盖,并且以i为根的树的最小点覆盖的大小。minSet[i][1]表示点i不属于点覆盖,并且以i为根的树的最小点覆盖的大小。
递归关系:
对于minSet[i][0],i的孩子结点可以属于点覆盖,也可以不属于点覆盖,取其使以i为根的子树的点覆盖最小的情况,因此
对于minSet[i][1],由于i不属于点覆盖,因此其所有孩子结点都必须属于点覆盖,因此
初值设定:
minSet[i][0]=1
minSet[i][1]=0
求解顺序:
从树的叶子结点开始求每个结点的最小点覆盖,自底向上,最后比较minSet[root][0]与minSet[root][1]的大小,最小者即为最终的结果。
- package org.xiu68.ch6.ex8;
- import java.util.ArrayList;
- public class Exp6_21 {
- public static void main(String[] args) {
- // TODO Auto-generated method stub
- //运行结果
- /*
- 树的最大独立集为: 4
- 顶点值为: 4 6 2 3
- 树的最小点覆盖为: 2
- 顶点值为: 5 1
- */
- //由结果可知 最大独立集与最小点覆盖集合互为补集
- ArrayList<Integer> vexs=new ArrayList<>();
- for(int i=1;i<=6;i++)
- vexs.add(i);
- //构造一个无向无环图
- int[][] edges=new int[][]{
- {0,1,1,0,0,0},
- {1,0,0,0,1,0},
- {1,0,0,0,0,0},
- {0,0,0,0,1,0},
- {0,1,0,1,0,1},
- {0,0,0,0,1,0}
- };
- MGraph<Integer> m=new MGraph<Integer>(6, 6, edges, vexs);
- m.maxIndependentSet();
- System.out.println();
- m.minCoverSet();
- }
- }
- //邻接矩阵表示图、无向无环图
- class MGraph<T>{
- public int vexNum; //顶点数量
- public int edgeNum; //边数量
- public int[][] edges; //邻接矩阵
- public ArrayList<T> vexs; //顶点表
- public int[][] maxDep; //最大独立集
- public ArrayList<Integer> set; //最大独立集顶点序号
- public int[][] minCover; //最小点覆盖
- public ArrayList<Integer> minSet; //最小点覆盖顶点序号
- public MGraph(int vexNum, int edgeNum, int[][] edges, ArrayList<T> vexs) {
- this.vexNum = vexNum;
- this.edgeNum = edgeNum;
- this.edges = edges;
- this.vexs = vexs;
- maxDep=new int[vexNum][2];
- set=new ArrayList<>();
- minCover=new int[vexNum][2];
- minSet=new ArrayList<>();
- }
- //最大独立集
- public void maxIndependentSet(){
- independentSet(0, 0);
- if(maxDep[0][0]>maxDep[0][1])
- System.out.println("树的最大独立集为: "+maxDep[0][0]);
- else
- System.out.println("树的最大独立集为: "+maxDep[0][1]);
- System.out.print("顶点值为: ");
- for(int i=0;i<set.size();i++)
- System.out.print(vexs.get(set.get(i))+" ");
- }
- //求以child为根的树的最大独立集
- //child:当前正在处理的结点
- //parent:child的父结点
- private void independentSet(int child,int parent){
- maxDep[child][0]=1; //当前结点放入独立集
- maxDep[child][1]=0; //当前结点不放入独立集
- for(int i=0;i<vexNum;i++){
- if(edges[child][i]==0 || i==parent) //如果顶点间不存在边或尾结点为父结点
- continue;
- independentSet(i, child);
- //因为child加入了最大独立集,所以子结点不加入最大独立集
- //以child为根的树的最大独立集的规模为 ( 1+ child的孙子结点的最大独立集的规模 )
- maxDep[child][0]+=maxDep[i][1];
- if(maxDep[i][0]>maxDep[i][1])
- maxDep[child][1]+=maxDep[i][0]; //加入子结点
- else
- maxDep[child][1]+=maxDep[i][1]; //不加入子结点
- }
- if(maxDep[child][0]>maxDep[child][1]) //比较加入child与不加入child的独立集大小,取较大者为结果
- set.add(child);
- }
- //***********************************************************
- //最小点覆盖
- public void minCoverSet(){
- coverSet(0,0);
- if(minCover[0][0]<minCover[0][1])
- System.out.println("树的最小点覆盖为: "+minCover[0][0]);
- else
- System.out.println("树的最小点覆盖为: "+minCover[0][1]);
- System.out.print("顶点值为: ");
- for(int i=0;i<minSet.size();i++){
- System.out.print(vexs.get(minSet.get(i))+" ");
- }
- }
- //求以child为根的树的最小点覆盖集合
- //child:当前正在处理的结点
- //parent:child的父结点
- private void coverSet(int child,int parent){
- minCover[child][0]=1; //child放入最小点覆盖集合
- minCover[child][1]=0; //child不放入最小点覆盖集合
- for(int i=0;i<vexNum;i++){
- if(edges[child][i]==0 || i==parent) //如果顶点间不存在边或尾结点为父结点
- continue;
- coverSet(i,child);
- //如果子结点i放入集合结果更小则把i放入集合
- if(minCover[i][0]<minCover[i][1])
- minCover[child][0]+=minCover[i][0]; //子结点i放入集合
- else
- minCover[child][0]+=minCover[i][1]; //子结点i不放入集合
- //若child不放入最小点覆盖集合,则其所有子结点都要放入最小点覆盖集合
- minCover[child][1]+=minCover[i][0];
- if(minCover[child][0]<minCover[child][1]) //取最小值作为结果
- minSet.add(child);
- }
- }
- }
Ex 6_21 最小点覆盖问题_第八次作业的更多相关文章
- Ex 6_14 布料剪裁问题_第八次作业
子问题定义: 定义p[i][j]为布料宽为i,高为j的最优产出,每次剪下一块布料,剩余布料最多形成三块矩阵面料.每次剪裁会有两种情况,水平切割布料,其次是将布料旋转90度后在切割布料. 递归关系: 初 ...
- Ex 6_5棋子放置问题_第八次作业
题目貌似有问题 (b) 子问题定义: 设maxValue[i][j]为棋盘的前i行中最后一行为i时第i行按照第j种放置方式放置时得到的最大覆盖值,comp[i][j]为第i种放置方式与第j种放置方式是 ...
- ACM/ICPC 之 机器调度-匈牙利算法解最小点覆盖集(DFS)(POJ1325)
//匈牙利算法-DFS //求最小点覆盖集 == 求最大匹配 //Time:0Ms Memory:208K #include<iostream> #include<cstring&g ...
- 【POJ 3041】Asteroids (最小点覆盖)
每次选择清除一行或者一列上的小行星.最少选择几次. 将行和列抽象成点,第i行为节点i+n,第j列为节点j,每个行星则是一条边,连接了所在的行列. 于是问题转化成最小点覆盖.二分图的最小点覆盖==最大匹 ...
- POJ 2226 最小点覆盖(经典建图)
Muddy Fields Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 8881 Accepted: 3300 Desc ...
- nyoj 237 游戏高手的烦恼 二分匹配--最小点覆盖
题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=237 二分匹配--最小点覆盖模板题 Tips:用邻接矩阵超时,用数组模拟邻接表WA,暂时只 ...
- [USACO2005][POJ2226]Muddy Fields(二分图最小点覆盖)
题目:http://poj.org/problem?id=2226 题意:给你一个字符矩阵,每个位置只能有"*"或者“.",连续的横着或者竖的“*"可以用一块木 ...
- POJ3041Asteroids(最小点覆盖+有点小抽象)
Asteroids Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 18289 Accepted: 9968 Descri ...
- hdu 1054 最小点覆盖
Sample Input 4 0:(1) 1 1:(2) 2 3 2:(0) 3:(0) 5 3:(3) 1 4 2 1:(1) 0 2:(0) 0:(0) 4:(0) Sample Output ...
随机推荐
- git中如何切换分支,拉取分支,合并分支
idea中如何使用git来做分支的切换合并: https://blog.csdn.net/autfish/article/details/52513465 本地分支与远程分支: https://seg ...
- H5 手机拨打电话与转到邮箱的标签属性
<a href="tel:电话号码"></a> <a href-"mailto:邮箱"></a> 说明:第一个标 ...
- String转换为Map
Map<String,Integer> rulsMap = new Gson().fromJson(cachedobj.toString(),new TypeToken<Map< ...
- java中各种时间公式
/** * 返回当前时间 * * @return 返回当前时间 */ public static Date getCurrentDateTime() { java.util.Calendar calN ...
- 面向对象【day08】:反射(五)
本节内容 概述 反射函数 综合使用 一.概述 反射我们以后会经常用到,这个东西实现了动态的装配,通过字符串来反射类中的属性和方法 二.反射函数 2.1 hasarttr(obj,name_str) 作 ...
- C#怎么调用百度地图Web API
直接上代码: public ActionResult FindMileage() { string s; HttpWebRequest req = (HttpWebRequest)HttpWebReq ...
- 在 chrome 上导出 pdf
用html+css写出网页,然后在chrome上导出pdf 1. command + p:快捷呼出打印: 2. “目标打印机”:选择“更改”,之后选择“另存为PDF”: 3. 点“更多设置”,可以勾选 ...
- JAVA记录-SpringMVC scope属性的两种模式
singleton作用域:当把一个Bean定义设置为singleton作用域是,Spring IoC容器中只会存在一个共享的Bean实例,并且所有对Bean的请求,只要id与该Bean定义相匹配,则只 ...
- Linux下删除命令 硬盘空间查看... 常用命令
(此命令请慎重使用) 使用rm -rf命令即可. 使用rm -rf 目录名字 命令即可 -r 就是向下递归,不管有多少级目录,一并删除-f 就是直接强行删除,不作任何提示的意思 (警告:不作任何 ...
- DEV Winform分页用户组件
资源部分在QQ群:616945527基于服务端数据分页,你也可以修改成本地分页.调用方法添加用户控件到窗体 public int curPage = 1;public int pageSize = 1 ...