子问题定义: 对于图中的每个结点,有两种状态,即属于最小点覆盖和不属于最小点覆盖,定义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]的大小,最小者即为最终的结果。

  1. package org.xiu68.ch6.ex8;
  2.  
  3. import java.util.ArrayList;
  4.  
  5. public class Exp6_21 {
  6.  
  7. public static void main(String[] args) {
  8. // TODO Auto-generated method stub
  9. //运行结果
  10. /*
  11. 树的最大独立集为: 4
  12. 顶点值为: 4 6 2 3
  13. 树的最小点覆盖为: 2
  14. 顶点值为: 5 1
  15. */
  16. //由结果可知 最大独立集与最小点覆盖集合互为补集
  17. ArrayList<Integer> vexs=new ArrayList<>();
  18. for(int i=1;i<=6;i++)
  19. vexs.add(i);
  20. //构造一个无向无环图
  21. int[][] edges=new int[][]{
  22. {0,1,1,0,0,0},
  23. {1,0,0,0,1,0},
  24. {1,0,0,0,0,0},
  25. {0,0,0,0,1,0},
  26. {0,1,0,1,0,1},
  27. {0,0,0,0,1,0}
  28. };
  29. MGraph<Integer> m=new MGraph<Integer>(6, 6, edges, vexs);
  30. m.maxIndependentSet();
  31. System.out.println();
  32. m.minCoverSet();
  33. }
  34. }
  35.  
  36. //邻接矩阵表示图、无向无环图
  37. class MGraph<T>{
  38. public int vexNum; //顶点数量
  39. public int edgeNum; //边数量
  40. public int[][] edges; //邻接矩阵
  41. public ArrayList<T> vexs; //顶点表
  42.  
  43. public int[][] maxDep; //最大独立集
  44. public ArrayList<Integer> set; //最大独立集顶点序号
  45.  
  46. public int[][] minCover; //最小点覆盖
  47. public ArrayList<Integer> minSet; //最小点覆盖顶点序号
  48.  
  49. public MGraph(int vexNum, int edgeNum, int[][] edges, ArrayList<T> vexs) {
  50. this.vexNum = vexNum;
  51. this.edgeNum = edgeNum;
  52. this.edges = edges;
  53. this.vexs = vexs;
  54.  
  55. maxDep=new int[vexNum][2];
  56. set=new ArrayList<>();
  57.  
  58. minCover=new int[vexNum][2];
  59. minSet=new ArrayList<>();
  60. }
  61.  
  62. //最大独立集
  63. public void maxIndependentSet(){
  64. independentSet(0, 0);
  65.  
  66. if(maxDep[0][0]>maxDep[0][1])
  67. System.out.println("树的最大独立集为: "+maxDep[0][0]);
  68. else
  69. System.out.println("树的最大独立集为: "+maxDep[0][1]);
  70.  
  71. System.out.print("顶点值为: ");
  72. for(int i=0;i<set.size();i++)
  73. System.out.print(vexs.get(set.get(i))+" ");
  74. }
  75. //求以child为根的树的最大独立集
  76. //child:当前正在处理的结点
  77. //parent:child的父结点
  78. private void independentSet(int child,int parent){
  79. maxDep[child][0]=1; //当前结点放入独立集
  80. maxDep[child][1]=0; //当前结点不放入独立集
  81.  
  82. for(int i=0;i<vexNum;i++){
  83. if(edges[child][i]==0 || i==parent) //如果顶点间不存在边或尾结点为父结点
  84. continue;
  85. independentSet(i, child);
  86.  
  87. //因为child加入了最大独立集,所以子结点不加入最大独立集
  88. //以child为根的树的最大独立集的规模为 ( 1+ child的孙子结点的最大独立集的规模 )
  89. maxDep[child][0]+=maxDep[i][1];
  90.  
  91. if(maxDep[i][0]>maxDep[i][1])
  92. maxDep[child][1]+=maxDep[i][0]; //加入子结点
  93. else
  94. maxDep[child][1]+=maxDep[i][1]; //不加入子结点
  95. }
  96.  
  97. if(maxDep[child][0]>maxDep[child][1]) //比较加入child与不加入child的独立集大小,取较大者为结果
  98. set.add(child);
  99. }
  100.  
  101. //***********************************************************
  102.  
  103. //最小点覆盖
  104. public void minCoverSet(){
  105. coverSet(0,0);
  106. if(minCover[0][0]<minCover[0][1])
  107. System.out.println("树的最小点覆盖为: "+minCover[0][0]);
  108. else
  109. System.out.println("树的最小点覆盖为: "+minCover[0][1]);
  110.  
  111. System.out.print("顶点值为: ");
  112. for(int i=0;i<minSet.size();i++){
  113. System.out.print(vexs.get(minSet.get(i))+" ");
  114. }
  115. }
  116. //求以child为根的树的最小点覆盖集合
  117. //child:当前正在处理的结点
  118. //parent:child的父结点
  119. private void coverSet(int child,int parent){
  120. minCover[child][0]=1; //child放入最小点覆盖集合
  121. minCover[child][1]=0; //child不放入最小点覆盖集合
  122.  
  123. for(int i=0;i<vexNum;i++){
  124. if(edges[child][i]==0 || i==parent) //如果顶点间不存在边或尾结点为父结点
  125. continue;
  126.  
  127. coverSet(i,child);
  128.  
  129. //如果子结点i放入集合结果更小则把i放入集合
  130. if(minCover[i][0]<minCover[i][1])
  131. minCover[child][0]+=minCover[i][0]; //子结点i放入集合
  132. else
  133. minCover[child][0]+=minCover[i][1]; //子结点i不放入集合
  134.  
  135. //若child不放入最小点覆盖集合,则其所有子结点都要放入最小点覆盖集合
  136. minCover[child][1]+=minCover[i][0];
  137.  
  138. if(minCover[child][0]<minCover[child][1]) //取最小值作为结果
  139. minSet.add(child);
  140. }
  141. }
  142. }

Ex 6_21 最小点覆盖问题_第八次作业的更多相关文章

  1. Ex 6_14 布料剪裁问题_第八次作业

    子问题定义: 定义p[i][j]为布料宽为i,高为j的最优产出,每次剪下一块布料,剩余布料最多形成三块矩阵面料.每次剪裁会有两种情况,水平切割布料,其次是将布料旋转90度后在切割布料. 递归关系: 初 ...

  2. Ex 6_5棋子放置问题_第八次作业

    题目貌似有问题 (b) 子问题定义: 设maxValue[i][j]为棋盘的前i行中最后一行为i时第i行按照第j种放置方式放置时得到的最大覆盖值,comp[i][j]为第i种放置方式与第j种放置方式是 ...

  3. ACM/ICPC 之 机器调度-匈牙利算法解最小点覆盖集(DFS)(POJ1325)

    //匈牙利算法-DFS //求最小点覆盖集 == 求最大匹配 //Time:0Ms Memory:208K #include<iostream> #include<cstring&g ...

  4. 【POJ 3041】Asteroids (最小点覆盖)

    每次选择清除一行或者一列上的小行星.最少选择几次. 将行和列抽象成点,第i行为节点i+n,第j列为节点j,每个行星则是一条边,连接了所在的行列. 于是问题转化成最小点覆盖.二分图的最小点覆盖==最大匹 ...

  5. POJ 2226 最小点覆盖(经典建图)

    Muddy Fields Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 8881   Accepted: 3300 Desc ...

  6. nyoj 237 游戏高手的烦恼 二分匹配--最小点覆盖

    题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=237 二分匹配--最小点覆盖模板题 Tips:用邻接矩阵超时,用数组模拟邻接表WA,暂时只 ...

  7. [USACO2005][POJ2226]Muddy Fields(二分图最小点覆盖)

    题目:http://poj.org/problem?id=2226 题意:给你一个字符矩阵,每个位置只能有"*"或者“.",连续的横着或者竖的“*"可以用一块木 ...

  8. POJ3041Asteroids(最小点覆盖+有点小抽象)

    Asteroids Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 18289   Accepted: 9968 Descri ...

  9. 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 ...

随机推荐

  1. git中如何切换分支,拉取分支,合并分支

    idea中如何使用git来做分支的切换合并: https://blog.csdn.net/autfish/article/details/52513465 本地分支与远程分支: https://seg ...

  2. H5 手机拨打电话与转到邮箱的标签属性

    <a href="tel:电话号码"></a> <a href-"mailto:邮箱"></a> 说明:第一个标 ...

  3. String转换为Map

    Map<String,Integer> rulsMap = new Gson().fromJson(cachedobj.toString(),new TypeToken<Map< ...

  4. java中各种时间公式

    /** * 返回当前时间 * * @return 返回当前时间 */ public static Date getCurrentDateTime() { java.util.Calendar calN ...

  5. 面向对象【day08】:反射(五)

    本节内容 概述 反射函数 综合使用 一.概述 反射我们以后会经常用到,这个东西实现了动态的装配,通过字符串来反射类中的属性和方法 二.反射函数 2.1 hasarttr(obj,name_str) 作 ...

  6. C#怎么调用百度地图Web API

    直接上代码: public ActionResult FindMileage() { string s; HttpWebRequest req = (HttpWebRequest)HttpWebReq ...

  7. 在 chrome 上导出 pdf

    用html+css写出网页,然后在chrome上导出pdf 1. command + p:快捷呼出打印: 2. “目标打印机”:选择“更改”,之后选择“另存为PDF”: 3. 点“更多设置”,可以勾选 ...

  8. JAVA记录-SpringMVC scope属性的两种模式

    singleton作用域:当把一个Bean定义设置为singleton作用域是,Spring IoC容器中只会存在一个共享的Bean实例,并且所有对Bean的请求,只要id与该Bean定义相匹配,则只 ...

  9. Linux下删除命令 硬盘空间查看... 常用命令

    (此命令请慎重使用) 使用rm -rf命令即可. 使用rm -rf 目录名字 命令即可 -r 就是向下递归,不管有多少级目录,一并删除-f 就是直接强行删除,不作任何提示的意思    (警告:不作任何 ...

  10. DEV Winform分页用户组件

    资源部分在QQ群:616945527基于服务端数据分页,你也可以修改成本地分页.调用方法添加用户控件到窗体 public int curPage = 1;public int pageSize = 1 ...