20172314 2018-2019-1《程序设计与数据结构》第九周学习总结

教材学习内容总结

无向图

  • 与树类似,由顶点和边连接构成。图只有顶点没有边时,也是图,是一种特殊形式。

  • 由相互连通的结点对表示,(A,B)表示从顶点A到顶点B有一条边,无序结点对表示顶点A,B之间的边可以从两个方向游历,边记做(A,,B)或(B,A)均可

  • 无向图就是边全为无序结点对的图

  • 两个顶点之间若有连通边,则称这两个顶点是邻接的。他们互为邻居

  • 路径表示一系列的边,路径的长度表示路径中边的条数(或顶点数减一);无向图中路径ABD与路径DBA等同。

  • 注意区分:如果无向图拥有最大数目的连通顶点的边,则无向图是完全的;如果无向图中任意两个顶点之间都存在一条路径,则认为无向图是连通的。如图是非连通无向图,D与另外三个顶点之间均没有任何路径。

  • 注意区分:联通一个节点及其自身的边称为自循环,边(A,A)表示连接A到自身的一个环;环路是一种首顶点和末顶点相同且没有重边的路径

  • 没有环路的图称为无环的

  • 无向树是一种连通的无环无向图,其中一个元素被指定为树根

有向图

  • 有向图也称双向图,是一种边为有序顶点对的图,边(A,B)和(B,A)是不同的有向边

  • 有向图的路径是连接两个顶点的有向边序列。路径ABD与路径DBA是不同的

  • 若有向图中任意两个顶点之间都存在一条路径,则认为该有向图是连通的。下图中第一个图是连通的,第二个图不是,因为没有任何路径能从其它顶点游历到1,也不能从6出发游历任何其他顶点

  • 如果有向图中没有环路,且有一条从A到B的边,那么就可以把A安排在顶点B之前,这种排列得到的顶点次序称为拓扑序

  • 有向树是一种指定了一个元素作为树根的有向图,该图具有如下属性:

    • 不存在其他顶点到树根的连接
    • 每个非树根元素恰好有一个连接
    • 树根到每个其它顶点都有一条路径

网络

  • 网络又称为加权图,是一种每条边都带有权重或代价的图
  • 加权图中的路径权重是该路径各边权重之和
  • 用三元组表示每条边,包括起始顶点、终止顶点和权重。对于有向图来说,必须包含每个有向连接的三元组

常用的图算法

  • 遍历

    • 广度优先遍历(BFS):类似于树的层次遍历,使用队列和无序列表
    • 深度优先遍历(DFS):类似于树的前序遍历,使用栈和无序列表
  • 测试连通性

    • 简单解释:在一个含有n个顶点的图中,当且仅当对每个顶点v,从v开始的广度优先遍历的resultList大小都是n,则该图就是连通的。例如

    无向连通图及以每个顶点为起点的广度优先遍历


    无向非连通图及以每个顶点为起点的广度优先遍历

  • 最小生成树

    • 生成树是一棵含有图中所有顶点和部分边(但可能不是所有边)的树。
    • 由于树也是图,有些图本身就是一棵生成树,这时该图的唯一生成树将包含所有边。
    • 最小生成树(保持图连通的最少边)(生成树不是唯一的)边的权重总和小于或等于同一图中其他任何一棵生成树的权重总和
    • 最小生成树集合包括n个点和n-1条边。且没有回路。
    • 最小生成树生成过程:从网络中任意选取一个起始顶点,并添加到最小生成树中,然后将所有含起始顶点的边按照权重次序添加到minheap中(如果处理的是有向网络,则只会添加那些以这个特定顶点为起点的边),接着从minheap中取出最小边,并与新顶点添加到最小生成树中。往minheap中添加所有含该新顶点且另一顶点尚不在最小生成树中的边。继续这一过程,直到最小生成树含有原始图中的所有顶点(或minheap为空)时结束。
  • 判断最短路径

    • 两顶点之间最小边数
      • 在编历时记录从起始顶点到本顶点的路径长度,以及路径中作为本顶点前驱的那个顶点。并当抵达目标顶点时循环终止,最短路径长度就是得到的路径长度加1
    • 加权图的最便宜路径
      • 使用minheap或优先队列来存储顶点,基于总权重衡量顶点对,每个顶点都必须存储该顶点的标签(本顶点之前最便宜的路径权重及路径上本顶点的前驱),从minheap取出顶点的时候,会权衡顶点对,如果未按由小到大取出顶点,则会更新路径

图的实现策略

  • 邻接列表

    • 用类似于链表的动态结点来存储每个结点带有的边。这种链表称为邻接列表
    • 对网络或加权图来说,每条边会存储成一个含权重的三元组;对无向图而言,边(A,B)会同时出现在顶点A和顶点B的邻接列表中
  • 邻接矩阵

    • 可以用集合储存边
    • 用称为邻接矩阵的二维数组存储顶点,每个单元表示两个顶点的交接情况,由表示是否连通的布尔值表示

    无向图的邻接矩阵

    因为是无向图,所以该矩阵沿对角线对称,只需给出一侧即可
    有向图的邻接矩阵

    邻接矩阵也可以用于网络或加权图,只需要在矩阵的各个单元中存储一个代表边权重的对象,没有边的单元为null

  • 用邻接矩阵实现无向图

    • addEdge方法:使用getIndex方法定位索引并调用addEdge方法进行赋值
    • addVertex方法:往图中添加一个顶点包括在数组的下一个可用位置添加该顶点,把邻接矩阵中所有恰当的位置都设置成false
    • expandCapacity方法:不仅扩展顶点数组并把已有顶点复制到新数组中,而且还必须扩展邻接列表的容量并把旧内容复制到新列表中

教材学习中的问题和解决过程

  • 问题一:广度优先遍历和深度优先遍历的实现过程及其原理

  • 问题一解决:通过查找资料其过程可认为:

    广度优先遍历(使用一个队列):

    • 首先选择一个顶点作为起始顶点,让起始顶点进入队列中,并将其染成灰色(visited),其余顶点为白色。
    • 开始循环:从队列首部选出一个顶点,添加到resultList末端(涂黑),并找出所有与之邻接的顶点,放入队列尾部(涂灰),没访问过的顶点是白色。然后再次取出新的起始顶点涂黑放入resultList中,如此循环。如果顶点的颜色是灰色,表示已经发现并且放入了队列,如果顶点的颜色是白色,表示还没有发现。
    • 基本就是出队的顶点变成黑色,在队列里的是灰色,还没入队的是白色。






    深度优先遍历(使用栈):

    • 其总体思想为:访问顶点v,依次从v的未被访问的邻接点出发,到一条路径的尽头时,将其入栈返回上一个顶点,若其有其他孩子,继续向下。直到所有顶点都被访问。

    过程为:

    • 先往栈中压入右节点,再压左节点,这样出栈就是先左节点后右节点了
    • 首先将顶点A压入栈中,stack(A)
    • 将A弹出,进入无序列表中,同时将v的右孩子B和左孩子C压入栈中,此时左孩子在栈的顶部,stack(B,C)
    • 将B顶点弹出,同时将B的子顶点E,D压入栈中,此时D在栈的顶部,stack(D,E,C)
    • 将D顶点弹出,没有子顶点压入,此时E在栈的顶部,stack(E,C)
    • 将E顶点弹出,同时将E的子顶点I压入,stack(I,C)
    • 依次往下,最终遍历完成

    两者之间唯一不同之处是:深度优先遍历使用栈而不是队列来管理遍历

  • 问题二:图的邻接列表较难理解

  • 问题二解决:邻接列表是对每个顶点建立一个链表,表示以改顶点为起点的所有顶点。通过访问一个顶点的链表即可得出该顶点的所有边。由此也可也得出以改顶点为起点的一条路径。
    有向图:


    一共有1234四个顶点,所以建立四个顶点,以1为起点的有234,以2为起点的只有4,以3为起点的没有,以4为起点的只有3。
    无向图:


    一共有ABCD四个顶点,建立4个链表。A顶点连接的边是BCD,与B顶点连接的是ABD,与C顶点连接的只有A,与D顶点连接的是AB。

代码调试中的问题和解决过程

  • 问题一:在iteratorBFS方法中,

     if (!indexIsValid(startIndex))
            return resultList.iterator();
    

    不清楚iterator方法的作用

  • 问题一解决:首先,这个方法的作用是为了使类可迭代,使类迭代需要有以下几步:

    • 在类声明中加入implements Iterable,对应的接口为(即java.lang.Iterator)

      public interface Iterable<Item>{
          Iterator<Item> iterator();
      }
      
    • 在类中实现iterator()方法,返回一个自己定义的迭代器Iterator

      public Iterator<Item> iterator(){
          //如果需要逆序遍历数组,自定义一个逆序迭代数组的迭代器
           return new ReverseArrayIterator();
      }
      
    • 在类中设置内部类(如private class ReverseArrayIterator() ),内部类声明中加入implements Iterator,对应的接口为(即java.util.Iterator)

      public interface Iterator {  
      

   boolean hasNext();
   Object next();
   void remove();
}
```
这些是在第四周中学到过的,比如ArrayList类,完全符合以上步骤。
在这里,由于索引无效,必须要返回一个相同类型的值,实际上就是空的。

  • 问题二:对广度优先遍历代码的理解

  • 问题二解决:见注释

    private Iterator<T> iteratorBFS(int startIndex) {
        Integer x;
        QueueADT<Integer> traversalQueue = new LinkedQueue<Integer>();
        UnorderedListADT<T> resultList = new ArrayUnorderedList<T>();
        //索引无效,返回空
        if (!indexIsValid(startIndex))
            return resultList.iterator();
        boolean[] visited = new boolean[maxCount];
        //把所有顶点设为false,白色
        for (int i = 0; i < maxCount; i++)
            visited[i] = false;
        //进入队列的为true,即访问过的,灰色
        traversalQueue.enqueue(startIndex);
        visited[startIndex] = true;
        while (!traversalQueue.isEmpty()) {
            //出队列涂黑存入resultList中
            x = traversalQueue.dequeue();
            resultList.addToRear((T) nodelist.get(x).getElement());
            //如果进入resultList的顶点还有相邻的未访问过的顶点,将其涂灰入队
            for (int i = 0; i < maxCount; i++) {
                if (hasEdge(x, i) && !visited[i]) {
                    traversalQueue.enqueue(i);
                    visited[i] = true;
                    Int++;
                }
            }
        }
        return new GraphIterator(resultList.iterator());
        }
    
  • 问题三:最短路径的代码理解,不知道iteratorShortestPathIndices方法和iteratorShortestPath方法的联系和区别。

  • 问题三解决:iteratorShortestPathIndices方法构建最短路径中顶点集合的迭代器,然后iteratorShortestPath方法获取迭代器并输出构成最短路径的结点。

    • iteratorShortestPath方法
    //最短路径的顶点集
    private Iterator<T> iteratorShortestPath(int startIndex, int targetIndex)
    {
        UnorderedListADT<T> resultList = new ArrayUnorderedList<T>();
        //如果索引值都无效,返回空
        if (!indexIsValid(startIndex) || !indexIsValid(targetIndex))
            return resultList.iterator();
        //it表示构成startindex和targetindex之间最短路径的顶点集,并存储在resultlist链表中,获取结点集合的迭代器对象
        Iterator<Integer> it = iteratorShortestPathIndices(startIndex, targetIndex);
        while (it.hasNext())
            resultList.addToRear((T)nodelist.get(((Integer)it.next())).getElement());
        return new GraphIterator(resultList.iterator());
    }
    
    • iteratorShortestPathIndices方法,构建从开始到结束的节点集合的迭代器对象
    //找到最短路径的顶点值
    private Iterator<Integer> iteratorShortestPathIndices(int startIndex, int targetIndex)
    {
        int index = startIndex;
        int[] pathLength = new int[maxCount];//路径长度数组
        int[] predecessor = new int[maxCount];//前驱结点
        QueueADT<Integer> traversalQueue = new LinkedQueue<Integer>();
        UnorderedListADT<Integer> resultList = new ArrayUnorderedList<Integer>();
        //如果索引无效或起始终点为同一索引,返回空
        if (!indexIsValid(startIndex) || !indexIsValid(targetIndex) || (startIndex == targetIndex))
            return resultList.iterator();
        boolean[] visited = new boolean[maxCount];//访问过为true,染灰
        //先标记为都没访问过,染白
        for (int i = 0; i < maxCount; i++)
            visited[i] = false;
        //将起始值入队标为访问过,染灰
        traversalQueue.enqueue(Integer.valueOf(startIndex));
        visited[startIndex] = true;
        pathLength[startIndex] = 0;//路径长度为0
        predecessor[startIndex] = -1;//前驱结点为-1位置
        //如果还有访问过的灰色,并且未找到目标索引
        while (!traversalQueue.isEmpty() && (index != targetIndex))
        {
            index = (traversalQueue.dequeue()).intValue();//出队列染黑,index储存其元素值
            //如果有其他结点与index有联系却没有被访问过的,入队染灰
            for (int i = 0; i < maxCount; i++)
            {
                if (hasEdge(index,i) && !visited[i])
                {
                    pathLength[i] = pathLength[index] + 1;//长度加一
                    predecessor[i] = index;//index为其前驱结点,predecessor中存储最短路径顶点
                    traversalQueue.enqueue(Integer.valueOf(i));//进队染灰
                    visited[i] = true;
                }
            }
        }
        //如果index不是目标索引,返回空
        if (index != targetIndex)
            return resultList.iterator();
        StackADT<Integer> stack = new LinkedStack<Integer>();
        index = targetIndex;
        stack.push(Integer.valueOf(index));//目标索引入栈
        do
        {//index不是起始索引值的元素值时,index表示其前驱结点的元素值,将结点的值依次入栈
            index = predecessor[index];
            stack.push(Integer.valueOf(index));
        } while (index != startIndex);
        while (!stack.isEmpty())
            resultList.addToRear(((Integer)stack.pop()));//栈不为空时,弹出结点添加到resultlist链表
        return new GraphIndexIterator(resultList.iterator());//resultlist中储存的就是最短路径的顶点集
    }
    
  • 问题四:出现从未遇到过的异常:Exception in thread "main" java.lang.OutOfMemoryError: Java heap space

  • 问题四解决:有三种可能导致此类错误的情况:

    • 真实的存在内存泄漏
    • 没有为应用程序提供足够多的内存,这种情况下,可以增加JVM堆可用空间大小,或者减少程序所需内存量
    • 程序中无心的对象引用保持,使得没有明确的释放对象,以致于堆增长再增长,直到没有额外的空间。
      网上的解决办法:

    选中被运行的类,点击菜单‘run->run...’,选择(x)=Argument标签页下的vm arguments框里
    输入 -Xmx800m, 保存运行。

    但我没怎么明白具体操作,我认为我的错误可能是第三种情况,于是把第一周中inkedStack方法检查了一下,发现方法size,isEmpty,和toString不完善,在其他方法的条件中调用时就有可能出错,修改之后问题解决。

代码托管

上周考试错题总结

结对及互评

  • 谭鑫20172305:谭鑫的博客中最突出的就是他的扩展学习很多,比如Tarjan算法是我没有了解的,可以说很用心了,博客总结的一直很详细,小小的建议就是希望解释的时候可以分一下层次,一大段一大段的有点难阅读(o´ω`o)ノ

  • 王禹涵20172323:王禹涵的博客整体很好,及哦啊才问题总结详细,有一个问题就是不太明白代码问题二。

其他

关于图的概念有逻辑性,容易理解,但是问题还是在代码理解上,我觉得Java学习不能停留在知识表面,要深入理解代码,并学会构造才是最终目的。

学习进度条

代码行数(新增/累积) 博客量(新增/累积) 学习时间(新增/累积)
目标 5000行 30篇 400小时
第一周 0/0 1/1 8/8
第二周 1163/1163 1/2 15/23
第三周 774/1937 1/3 12/50
第四周 3596/5569 2/5 12/62
第五周 3329/8898 2/7 12/74
第六周 4541/13439 3/10 12/86
第七周 1740/15179 1/11 12/97
第八周 5947/21126 1/12 12/109
第九周 7968/29094 2/14 12/121

参考

posted @ 2018-11-18 11:11  FYiW  阅读(213)  评论(0编辑  收藏  举报