20172305 2018-2019-1 《Java软件结构与数据结构》第八周学习总结

教材学习内容总结

本周内容主要为书第十二章内容:

  • (附加属性的二叉树)

    • 完全二叉树
    • (最小堆)对于每一个结点,它小于或等于其左孩子和右孩子。
    • (最大堆)对于每一个结点,它大于或等于其左孩子和右孩子。


  • 最小堆将其最小元素存储在二叉树的根处,其根的两个孩子同样也是最小堆。
  • 最大堆将其最大元素存储在二叉树的根处,其根的两个孩子同样也是最大堆。

  • addElement方法将给定的元素添加到堆中的恰当位置处,且维持该堆的完全性属性和有序属性。
    • 如果给定元素不是Comparable的,则该方法将会抛出一个ClassCastException异常。
    • 对于新插入结点位置要么是第h层左边的下一个空位置,要么是第h+1层左边的第1个位置(h层满的话)。
    • 考虑排序属性,将该新值和双亲值进行比较,如果新结点小于双亲则互换位置,沿着树向上继续此过程,直至该新结点要么是大于双亲要么是位于堆的根处。

      添加元素0:

  • removeMin方法将删除最小堆中的最小元素并返回它。
    • 由于删除元素是存储在最小堆的根处,所以要返回存储在根处元素并用堆中的最末一片叶子上的元素替代。一旦最末一片叶子的元素移动到根处,则必须要对堆进行重新排序以维持堆的排序属性。
    • 新根元素与较小的子结点进行比较,如果子结点更小就替换,沿着树向下这一过程,直至该元素要么位于某片叶子要么比两个子结点都小。


  • findMin方法将返回一个指向最小堆中的最小元素的引用,只需通过返回存储在根处的元素。

  • 三种方法的时间复杂度:

  • 堆--优先级队列(遵循两个排序规则的集合,具有更高级的项目在先,具有相同优先级的项目使用先进先出的方法来确定其排序)

  • 链表实现堆:
    • addElement操作使用了两个私有方法getNextParentAdd和heapifyAdd两个方法,getNextParentAdd方法是返回一个指向某结点的引用,该结点为插入结点的双亲。heapifyAdd方法是完成堆的任何重排序,从那片新叶子开始,向上处理至根处。

      • 在恰当位置处添加一个新元素,对堆进行重排序来维持排序属性,将lastNode指针重新设定为指定新的最末结点。
    • removeMin操作使用了两个私有方法getNewLastNode和heapifyRemove两个方法,getNewLastNode方法返回一个指向某一结点的引用,该结点将是新的最末结点。heapifyRemove方法将完成任何有必要的树重排序,从根向下进行。

      • 用存储在最末结点处的元素替换存储在根处的元素,对堆进行重排序,以及返回初始的根元素。
    • findMin方法返回一个指向存储在根结点的元素的内容。

  • 数组实现堆:
    • addElement操作使用了私有方法heapifyAdd方法,用于在必要时对该堆进行重组。

      • 在恰当位置处添加新结点,对堆进行重排序来维持排序属性,将count递增。
    • removeMin操作使用了私有方法heapifyRemove方法,用于在必要时对该堆进行重组。

      • 用存储在最末结点处的元素替换存储在根处的元素,对堆进行重排序,以及返回初始的根元素。
    • findMin方法返回一个指向存储在根结点的元素的内容。

  • 堆排序

    • 将列表的每一个元素添加到堆中,然后在一次一个的将他们从根中删除。
    • 最小堆的情况下,排序结果是将该列表以升序排列
    • 最大堆的情况下,排序结果是将该列表以降序排列
    • 时间复杂度:O(nlogn)
    • 空间复杂度:O(1)

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

  • 问题1:实现优先级队列的二叉堆、d堆、左式堆、斜堆、二项堆都是什么意思?和最小堆、最大堆有什么区别?
  • 问题1解决方案:先说后面的问题,最小堆和最大堆是二叉堆的两种表现形式,最大堆是除了根以外的所有结都要满足:父结点大于子结点;而最小堆恰恰相反,最小堆是除了根以外的所有结都要满足:父结点小于子结点。所以,二叉堆可以根据优先级确定出队列的先后顺序。

    • d堆是完全d叉树,就是每一个结点有d个子结点,同时还要满足每个结点从左到右的顺序。
    • 左式堆是具有堆序性质的二叉树,左式堆的任意结点的值比其子树的任意结点值均小,但和一般的二叉堆不同,左式堆不再是一棵完全二叉树。
    • 斜堆也叫自适应堆,是一种使用二叉树实现的堆状数据结构,使用的是二叉树而不是完全二叉树,所以在整体来看,斜堆会是极不平衡的一个堆,但其合并的速度远远大于二叉堆。
    • 二项堆是是二项树的集合或是由一组二项树组成,在O(logn)的时间内即可完成两个二项堆合并操作,所以二项堆是可合并堆,基于二项堆实现的优先队列和进程调度算法有着很好的时间性能。

代码学习中的问题和解决过程

  • 问题1:如何解决数组实现的堆在删除操作之后会输出被删元素?分析数组实现删除添加操作的方法?
  • 问题1的解决方案:删除一个元素之后,进行整体的输出会出现被删除元素,通过对代码的分析,在删除最小结点后将末叶结点放到根处之后,没有将叶结点内容进行删除,而是在计数变量和操作次数进行递减,这样的话叶结点和暂时的根结点树都会有同一元素。所以在tree[0]=tree[count - 1];heapifyRemove()之间添加一句tree[count - 1] = null;直接令其为空。针对删除时对堆的排序方法(向下遍历),先判断根结点的左侧和右侧是否为空,如果为空的话就直接跳过循环让新的根结点为根结点,如果不为空的话会有两种情况,只有一个子结点,此时右侧为空,则左侧不为空,记录此时左结点的索引值,如果左右都不为空的话进行判断左侧和右侧谁更小(最小堆)。进入循环,循环条件是索引值小于总共数量和索引值位置与新根结点的比较,node始终是next的父结点的索引值,不断向下遍历,而对于每一个结点他的索引值是n的话,其左子结点为2n+1,右子结点为2(n+1),和之前操作类似,判断来年两个子结点的大小,向下不断遍历,next的值赋给node,并找node的孩子的索引,令next重新指向node的孩子直至不满足next索引值超过总的或是元素大小不符合的时候会跳出循环,node的位置就是替代新结点的位置。针对添加时对堆的排序方法(向上排序),添加的方法进行的排序比较简单,直接确定添加的位置是否比父结点小,如果小的话,进行调换;大得话跳出循环结束排序。和之前的类似,子结点的索引值为count-1的话,那么其父结点位置为next-1/2,不断的和父结点进行比较直至向上遍历结束,当跳出循环后,此时的next位置就是新添结点的位置。

    未加tree[count - 1] = null

    添加之后:

  • 问题2:PP12.1如何让用堆来实现队列?队列的方法如下:
  • 问题2的解决方案:用堆来实现队列的问题,因为队列是一个线性结构,先进先出,而堆是完全二叉树。所以,需要根据把每一个进行队列的元素标上号,按照号码牌的内容来实现堆的形式,但是如果这样的话,我们就不能直接往堆里面插入泛型元素,需要定义个类,而在这个类的内部需要每定义一个就会自增数量,比如先进的是A元素,就为1;再进的是B元素,就为2...以此类推。类似的形式在实验室内就帮助学长实现过,所以定义类个类并不难。但是,需要注意的是要在类中写出比较方法,确定比较元素是谁。但是,我第一次实现的过程中,将这个计数的比较放在外部进行,这样实现的话可以确定谁先进对谁先出队。但是在实现输出队列内容的时候,会发现输出的内容不是进队的形式。而是中序的形式(问题3会解决此问题)。但是,我发现队列的输出形式可以每次删除堆的根结点,用一个暂时变量存放内容,这样的话就复刻了一份堆,利用了复刻的堆实现输出而保持了原堆的内容。但是在实现这种情况会始终出现两个堆被删的一干二净,虽然会按照队列的形式进行输出,但是往后删除就会有问题。经过改写,尝试把删除元素存放到暂时变量中,在给它塞回去。这种方法虽然实现了不报错,但是始终会输出第一个元素,其他元素不输出。我认为是因为我规定的存放数字是第一个就永远第一个。通过侯泽洋同学的交流,发现我的代码都是复制粘贴,没用到继承,如果用继承的话就直接调用父类的方法就可以了,而且规定的标记数字也可以写进方法里面,不用自己规定,保持了队列的完整性。但是,这样的话我还是只输出第一个元素。后来感觉如归的位置重新定义一下,就运行成功了,可能和第一次犯错一样就是在再次入队的时候他的标记数字仍是之前的就会一直输出,重新定义一下就会将之前的标记数字抹去换新。

    进行复刻导致两个堆的元素都被删除:


    实现队列输出(只是第一个元素的输出):


    真正实现队列输出:

  • 问题3:如何改写数组实现堆的输出形式?
  • 问题3的解决方案:这个问题是在课堂测试上发现的,但是我在ArrayHeap方法里面并没有找到toString方法,但是调用toString没有问题。后来通过单步调试跳到了ArrayBinary的toString。通过继承会调用父类的方法,如过在子类中重写方法会调用子类的方法。链表实现的堆也是如此。所以,针对toString方法我们可以有两种修改方法,一是修改子类重写,二是修改父类直接调用父类的。

代码托管

上周考试错题总结

  • 第九章:
  • 错题1:Insertion sort is an algorithm that sorts a list of values by repetitively putting a particular value into its final, sorted, position.
    • A .true
    • B .false
  • 错误解析:插入排序是一种反复的把某个元素插入到之前已经排好序的列表中。

  • 错题2:The best comparison sort in terms of order is:

    • A .O(1)
    • B .O(n)
    • C .O(log(n))
    • D .O(nlog(n))
  • 错误解析:根据各个排序方法的比较快速和归并排序都是O(nlogn)的时间复杂度,运行效率高。

  • 第十章:
  • 错题3:The Java Collections API provides two implementations of balanced binary search trees, TreeSet and TreeMap, both of which use a ___________tree implementation.
    • A .AVL
    • B .red/black
    • C .binary search
    • D .None of the above
  • 错误解析:通过实验二的第六部分学习,知道两个类都是红黑树。

  • 第十一章:
  • 错题4:A minheap stores its largest element at the root of the binary tree, and both children of the root of a minheap are also minheaps.
    • A .True
    • B .Flase
  • 错误解析:最小堆将其最小元素存放到根处,不是最大元素,最大堆将其最大元素存放到根处。

  • 错题5:Since a heap is a binary search tree, there is only one correct location for the insertion of a new node, and that is either the next open position from the left at level h or the first position on the left at level h+1 if level h is full.

    • A .True
    • B .Flase
  • 错误解析:堆是一个完全树,不是二叉查找树。

  • 三章的错题每次测试都差一点点满分,而且在写此部分的过程中发现每道题都是学过的边边角角之类的问题。暴漏出学习的不足,看书还是不够仔细,针对边界类的问题没有仔细思考。

结对与互评

点评(王禹涵)

  • 博客中值得学习的或问题:

    • 教材总结写的很好,排版整齐。盗图还被称为丑图有点不太合适吧。问题2的排版可以再调整一下会更好的。
  • 代码中值得学习的或问题:
    • 迭代器的问题可以推荐看看第七章的内容,迭代器的输出方式有两种方法可以都去尝试一下!
  • 基于评分标准,我给本博客打分:7分。
    • 得分情况如下:
    • 正确使用Markdown语法(加1分)
    • 模板中的要素齐全(加1分)
    • 教材学习中的问题和解决过程, 二个问题加2分
    • 代码调试中的问题和解决过程, 一个问题加1分
    • 感想,体会不假大空的加1分
    • 点评认真,能指出博客和代码中的问题的加1分

点评(方艺雯)

  • 博客中值得学习的或问题:

    • 图片排版不错,问题二的解决方案的文字部分可以稍稍修饰一下,堆排序的部分写的很好很细致,配图也很精细的把每一次结果都放上去了。建议可以更细致化的提出问题。
  • 代码中值得学习的或问题:
    • 代码中的问题链接了我的博客,有点受宠若惊,总结的很详细。
  • 基于评分标准,我给本博客打分:11分。
  • 得分情况如下:
    • 正确使用Markdown语法(加1分)
    • 模板中的要素齐全(加1分)
    • 教材学习中的问题和解决过程, 四个问题加4分
    • 代码调试中的问题和解决过程, 三个问题加3分
    • 感想,体会不假大空的加1分
    • 点评认真,能指出博客和代码中的问题的加1分

互评对象

感悟

第十二章的优先队列与堆的内容并不是很难,和第十一章类似,都是在树的基础上添加附加条件,所以并不是很难,而且有很多示例代码可以学习,这样的话更减少困难。

学习进度条

代码行数(新增/累积) 博客量(新增/累积) 学习时间(新增/累积) 重要成长
目标 5000行 30篇 400小时
第一周 0/0 1/1 15/15
第二周 703/703 1/2 20/35
第三周 762/1465 1/3 20/55
第四周 2073/3538 1/4 40/95
第五周 981/4519 2/6 40/135
第六周 1088/5607 2/8 50/185
第七周 1203/6810 1/9 50/235
第八周 2264/9074 2/11 50/285

参考资料

20172305 2018-2019-1 《Java软件结构与数据结构》第八周学习总结的更多相关文章

  1. 20172328 2018—2019《Java软件结构与数据结构》第二周学习总结

    20172328 2018-2019<Java软件结构与数据结构>第二周学习总结 概述 Generalization 本周学习了第三章集合概述--栈和第四章链式结构--栈.主要讨论了集合以 ...

  2. 20172305 2018-2019-1 《Java软件结构与数据结构》第二周学习总结

    20172305 2018-2019-1 <Java软件结构与数据结构>第二周学习总结 教材学习内容总结 本周内容主要为书第三章和第四章的内容: 第三章(以数组来替代栈的作用) 集合(聚集 ...

  3. 20172305 2018-2019-1 《Java软件结构与数据结构》第九周学习总结

    20172305 2018-2019-1 <Java软件结构与数据结构>第九周学习总结 教材学习内容总结 本周内容主要为书第十五章内容: 图(结点和结点之间的连接构成) 顶点:结点 边:结 ...

  4. 20172305 2018-2019-1 《Java软件结构与数据结构》第一周学习总结

    20172305 2018-2019-1 <Java软件结构与数据结构>第一周学习总结 教材学习内容总结 本周内容主要为书第一章和第二章的内容: 第一章 软件质量: 正确性(软件达到特定需 ...

  5. 20172328 2018-2019《Java软件结构与数据结构》第一周学习总结

    20172328 2018-2019<Java软件结构与数据结构>第一周学习总结 概述 Generalization 本周学习了软件质量.数据结构以及算法分析的具体内容,主要依托于所用教材 ...

  6. 20172328 2018-2019《Java软件结构与数据结构》第九周学习总结

    20172328 2018-2019<Java软件结构与数据结构>第九周学习总结 概述 Generalization 本周学习了无向图.有向图.带权图.常用的图算法.图的实现策略. 教材学 ...

  7. 2018-2019-20172329 《Java软件结构与数据结构》第九周学习总结

    2018-2019-20172329 <Java软件结构与数据结构>第九周学习总结 教材学习内容总结 <Java软件结构与数据结构>第十五章-图 一.图及无向图 1.图的相关概 ...

  8. 2018-2019-20172321 《Java软件结构与数据结构》第九周学习总结

    2018-2019-20172321 <Java软件结构与数据结构>第九周学习总结 教材学习内容总结 第15章 图 无向图 图由顶点和边组成. 顶点由名字或标号来表示,如:A.B.C.D: ...

  9. 《JAVA软件结构与数据结构》第一周学习总结

    学号 20172326 <JAVA软件结构与数据结构>第一周学习总结 教材学习内容总结 软件质量的几大特性 增长函数与大O记法 大O记法用来表示表示增长函数,从而来表示算法的复杂度 算法的 ...

  10. 2018-2019-20172329 《Java软件结构与数据结构》第二周学习总结

    2018-2019-20172329 <Java软件结构与数据结构>第二周学习总结 教材学习内容总结 <Java软件结构与数据结构>第三章 集合概述--栈 一.集合 1.我们印 ...

随机推荐

  1. MySQL学习【第七篇索引管理及执行计划】

    一.索引介绍 1.什么是索引? 索引由如字典,目的就是为了更快寻找到要找的内容. 令搜索查询的数据更有目的性,从而提高数据检索的能力 2.索引类型介绍 1.BTREE: B+树索引 2.HASH: H ...

  2. vue---class和style的基本用法

    不多BB了 直接上代码了 通俗移动易懂总结了5种常用改变样式 的形式 <style> .actived2{ color:red; } </style> </head> ...

  3. 一个数据仓库时代开始--Hive

    一.什么是 Apache Hive? Apache Hive 是一个基于 Hadoop Haused 构建的开源数据仓库系统,我们使用它来查询和分析存储在 Hadoop 文件中的大型数据集.此外,通过 ...

  4. 一条SQL语句的千回百转

    SQL语言相信大家都不陌生,从本质上来说,它是一种结构化查询语言,是用来数据库之间的通信的编程语言.作为一名Java程序员,我们从Java角度来看,SQL语言相当于Java接口,而数据库是实现这个接口 ...

  5. 基于 FPGA 的 PCIE 总线 Linux 驱动设计

    硬件平台 Kintex ®-7 family of FPGAs Intel X86 软件平台 Linux 4.15.0-36-generic #39~16.04.1-Ubuntu Xilinx xap ...

  6. DDoS 攻击与防御:从原理到实践

    本文来自 网易云社区 . 可怕的 DDoS 出于打击报复.敲诈勒索.政治需要等各种原因,加上攻击成本越来越低.效果特别明显等趋势,DDoS 攻击已经演变成全球性的网络安全威胁. 危害 根据卡巴斯基 2 ...

  7. 参考 https://raspberrypi.stackexchange.com/questions/3617/how-to-install-unrar-nonfree > 1.卸载unrar-free。 $ sudo apt-get remove unrar-free \ 2.通过编辑确保您拥有源存储库/etc/apt/sources.list。 $ cat /etc/apt/sources.

    from my CSDN: https://blog.csdn.net/su_cicada/article/details/86939944 参考 https://raspberrypi.stacke ...

  8. hadoop errors

    1.taskTracker和jobTracker 启动失败 2011-01-05 12:44:42,144 ERROR org.apache.hadoop.mapred.TaskTracker: Ca ...

  9. MyOD课堂实践(5月31日)20155318

    MyOD课堂实践(5月31日)20155318 编写MyOD.java 用java MyOD XXX实现Linux下od -tx -tc XXX的功能 (码云链接) 代码 import java.io ...

  10. Cocoa Touch提供了哪几种Core Animation过渡类型?

    过渡动画通过 type 设置不同的动画效果, CATransition 有多种过渡效果, 但其实 Apple 官方的SDK只提供了四种: fade 淡出 默认 moveIn 覆盖原图 push 推出 ...