課程:《Java軟件結構與數據結構》
班級: 1723
姓名: 郭愷
學號:20172301
實驗教師:王志強老師
實驗日期:2018年11月20日
必修/選修: 必修html
LinkedBinaryTree
由於是以前的程序項目,因此實現起來很容易。java
getRight()
方法,首先在LinkedBinaryTree
類裏面聲明一個全局變量protected LinkedBinaryTree<T> left,right;
而後在構造函數裏,添加下面兩行代碼。node
// 建立以指定元素爲根元素的二叉樹,把樹做爲它的左子樹和右子樹 public LinkedBinaryTree(T element, LinkedBinaryTree<T> left, LinkedBinaryTree<T> right) { root = new BinaryTreeNode<T>(element); root.setLeft(left.root); root.setRight(right.root); this.left = left; this.right = right; }
而後直接返回right
便可。編程
// 返回此樹的根的右子樹。 public LinkedBinaryTree<T> getRight() { return right; }
contains
方法基於私有方法findAgain
實現。只須要判斷在樹裏可否找到目標元素便可。public boolean contains(T targetElement) { return findAgain(targetElement, root) != null; }
toString
方法,這裏爲了讓輸出是一個樹型,我用了以前ExpressionTree
的printTree
方法。一樣,toString
方法能夠考慮使用四種遍歷方式。preorder
方法和postorder
方法,實現理念和書上給的中序遍歷同樣。只須要調整一下左右孩子還有結點的順序。// 執行遞歸先序遍歷。 protected void preOrder(BinaryTreeNode<T> node, ArrayUnorderedList<T> tempList) { if (node != null) { tempList.addToRear(node.getElement()); preOrder(node.getLeft(), tempList); preOrder(node.getRight(), tempList); } }
// 執行遞歸後序遍歷。 protected void postOrder(BinaryTreeNode<T> node, ArrayUnorderedList<T> tempList) { if (node != null) { postOrder(node.getLeft(), tempList); postOrder(node.getRight(), tempList); tempList.addToRear(node.getElement()); } }
HDIBEMJNAFCKGL
和先序ABDHIEJMNCFGKL
。A
是樹的根結點。那麼根據中序得知A
的左邊是左子樹,A
的右邊是右子樹。B
便是左子樹的根,根據中序得知B
的左邊是左子樹的左孩子,B
的右邊是左子樹的右孩子。A
爲根的樹。A
爲根結點和遞歸實現的左右子樹,建立一個新樹。ExpresstionTree
是關於後綴表達式計算的一個樹。findMin
和findMax
只要分別查找最左結點和最右結點便可。在看源代碼以前,我以爲有必要學習一下如何去系統的看程序源代碼。這裏作一些摘錄:
第一,找准入口出口,不要直接跳進去看,任何代碼都有觸發點,不管是http request,仍是服務器自動啓動,仍是main函數,仍是其餘的,先從入口開始。
第二,手邊一支筆一張紙,除非你是Jeff,不然你不會記得那麼多跳轉的。一個跳轉就寫下來函數/方法名和參數,讀完一遍,就有了一個sequence diagram雛形 。
第三,私有方法掠過,只要記住輸入輸出便可,無需看具體實現數組
紅黑樹遵循如下五點性質:
性質1 結點是紅色或黑色。
性質2 根結點是黑色。
性質3 每一個葉子結點(NIL結點,空結點)是黑色的。
性質4 每一個紅色結點的兩個子結點都是黑色。(從每一個葉子到根的全部路徑上不能有兩個連續的紅色結點)
性質5 從任一結點到其每一個葉子結點的全部路徑都包含相同數目的黑色結點。服務器
TreeMap
聲明public class TreeMap<K,V> extends AbstractMap<K,V> implements NavigableMap<K,V>, Cloneable, java.io.Serializable {
public TreeMap() { comparator = null; }
無參構造方法,不指定比較器,排序的實現要依賴key.compareTo()
方法,所以key
必須實現Comparable
接口,並覆寫其中的compareTo
方法。
public TreeMap(Comparator<? super K> comparator) { this.comparator = comparator; }
採用帶比較器的構造方法,排序依賴該比較器,key
能夠不用實現Comparable
接口。
public TreeMap(Map<? extends K, ? extends V> m) { comparator = null; putAll(m); }
構造方法一樣不指定比較器,調用putAll
方法將Map
中的全部元素加入到TreeMap
中。
public TreeMap(SortedMap<K, ? extends V> m) { comparator = m.comparator(); try { buildFromSorted(m.size(), m.entrySet().iterator(), null, null); } catch (java.io.IOException | ClassNotFoundException cannotHappen) { } }
將比較器指定爲m的比較器,然後調用buildFromSorted方法,將SortedMap中的元素插入到TreeMap中,根據SortedMap建立的TreeMap,將SortedMap中對應的元素添加到TreeMap中。數據結構
public V put(K key, V value) { //獲得紅黑樹根結點 Entry<K,V> t = root; if (t == null) { compare(key, key); // type (and possibly null) check // 若是樹爲空,新建紅黑樹根結點 root = new Entry<>(key, value, null); size = 1; modCount++; return null; } //若是Map不爲空,找到插入新節點的父節點 int cmp; Entry<K,V> parent; Comparator<? super K> cpr = comparator; // 若是比較器不爲空 if (cpr != null) { do { // 使用 parent 上次循環後的 t 所引用的 Entry parent = t; // 拿新插入的key和t的key進行比較 cmp = cpr.compare(key, t.key); // 若是新插入的key小於t的key,t等於t的左結點 if (cmp < 0) t = t.left; // 若是新插入的key大於t的key,t等於t的右結點 else if (cmp > 0) t = t.right; else // 若是兩個key相等,新value覆蓋原有的value,並返回原有的value return t.setValue(value); } while (t != null); } // 沒有提供比較器 else { if (key == null) throw new NullPointerException(); @SuppressWarnings("unchecked") Comparable<? super K> k = (Comparable<? super K>) key; do { parent = t; cmp = k.compareTo(t.key); if (cmp < 0) t = t.left; else if (cmp > 0) t = t.right; else return t.setValue(value); } while (t != null); } // 將新插入的結點做爲parent結點的子結點 Entry<K,V> e = new Entry<>(key, value, parent); // 做爲左孩子 if (cmp < 0) parent.left = e; // 做爲右孩子 else parent.right = e; // 修復紅黑樹 fixAfterInsertion(e); size++; modCount++; return null; }
HashMap
類的聲明,能夠了解他繼承了什麼類和實現了哪些接口。public class HashMap<K,V> extends AbstractMap<K,V> implements Map<K,V>, Cloneable, Serializable {
<K,V>應該表示的是一種映射關係。app
HashMap 的實例有兩個參數影響其性能:初始容量 和加載因子。
容量是哈希表中桶的數量,初始容量只是哈希表在建立時的容量。
加載因子是哈希表在其容量自動增長以前能夠達到多滿的一種尺度。當哈希表中的條目數超出了加載因子與當前容量的乘積時,則要對該哈希表進行 rehash 操做(即重建內部數據結構),從而哈希表將具備大約兩倍的桶數。函數
這裏有兩個比較重要的變量,容量和加載因子。容量的值是2的n次冪,加載因子默認爲0.75。
源碼分析
這裏加載因子爲何默認是0.75呢?
一般,默認加載因子 (0.75) 在時間和空間成本上尋求一種折衷。加載因子太高雖然減小了空間開銷,但同時也增長了查詢成本(在大多數 HashMap 類的操做中,包括 get 和 put 操做,都反映了這一點)。在設置初始容量時應該考慮到映射中所需的條目數及其加載因子,以便最大限度地減小 rehash 操做次數。
若是初始容量大於最大條目數除以加載因子,則不會發生 rehash 操做。
這裏的rehash操做,我以爲相似於數組的擴容。
加載因子就是表示數組鏈表填滿的程度。
加載因子越大,填滿的元素越多,空間利用率高了,但衝突的機會加大了.鏈表長度會愈來愈長,查找效率下降。
加載因子越小,填滿的元素越少,衝突的機會減少了,但空間浪費多了.表中的數據將過於稀疏,不少空間還沒用,就開始擴容了。
// 初始容量(必須是2的n次冪),負載因子 public HashMap(int initialCapacity, float loadFactor) { if (initialCapacity < 0) throw new IllegalArgumentException("Illegal initial capacity: " + initialCapacity); if (initialCapacity > MAXIMUM_CAPACITY) initialCapacity = MAXIMUM_CAPACITY; if (loadFactor <= 0 || Float.isNaN(loadFactor)) throw new IllegalArgumentException("Illegal load factor: " + loadFactor); this.loadFactor = loadFactor; this.threshold = tableSizeFor(initialCapacity); }
HashMap
使用了數組,鏈表和紅黑樹,多種實現。static class Node<K,V> implements Map.Entry<K,V> { final int hash; final K key; V value; Node<K,V> next; Node(int hash, K key, V value, Node<K,V> next) { this.hash = hash; // 哈希值 this.key = key; // 鍵 this.value = value; // 值 this.next = next; // 下一個 } public final K getKey() { return key; } public final V getValue() { return value; } public final String toString() { return key + "=" + value; } public final int hashCode() { return Objects.hashCode(key) ^ Objects.hashCode(value); } public final V setValue(V newValue) { V oldValue = value; value = newValue; return oldValue; } // 判斷兩個Node是否相等 // 若兩個Node的「key」和「value」都相等,則返回true。 // 不然,返回false public final boolean equals(Object o) { if (o == this) return true; if (o instanceof Map.Entry) { Map.Entry<?,?> e = (Map.Entry<?,?>)o; if (Objects.equals(key, e.getKey()) && Objects.equals(value, e.getValue())) return true; } return false; } }
next
,是用來處理衝突的。HashMap
原本就是一個數組,若是數組中某一個索引起生了衝突,那麼就會造成鏈表。而鏈表到必定程度的時候,就會造成紅黑樹。
put
操做public V put(K key, V value) { return putVal(hash(key), key, value, false, true); } final V putVal(int hash, K key, V value, boolean onlyIfAbsent, boolean evict) { Node<K,V>[] tab; Node<K,V> p; int n, i; if ((tab = table) == null || (n = tab.length) == 0) // resize()方法是從新調整HashMap的大小 n = (tab = resize()).length; // 若不爲null,計算該key的哈希值,而後將其添加到該哈希值對應的鏈表中 if ((p = tab[i = (n - 1) & hash]) == null) tab[i] = newNode(hash, key, value, null); else { Node<K,V> e; K k; if (p.hash == hash && ((k = p.key) == key || (key != null && key.equals(k)))) e = p; //若是是紅黑樹結點 else if (p instanceof TreeNode) e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value); else { for (int binCount = 0; ; ++binCount) { if ((e = p.next) == null) { p.next = newNode(hash, key, value, null); if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st treeifyBin(tab, hash); break; } if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k)))) break; p = e; } } if (e != null) { // existing mapping for key V oldValue = e.value; if (!onlyIfAbsent || oldValue == null) e.value = value; afterNodeAccess(e); return oldValue; } } ++modCount; if (++size > threshold) resize(); afterNodeInsertion(evict); return null; }
問題1:實驗二基於(中序,先序)序列構造惟一一棵二㕚樹的功能,出現了ArrayIndexOutOfBoundsException
異常,簡單來講就是數組爲空。
如圖
根據調試發現,在新建了子樹的中序和先序數組以後,我並無把原來數組的值導入到新的數組當中。因此致使運行時拋出了數組空的異常。
如圖
for (int y = 0; y < in.length; y++) { // 把原數組的數存入新的數組當中 if (y < x) { inLeft[y] = in[y]; preLeft[y] = pre[y + 1]; } else if (y > x) { inRight[y - x - 1] = in[y]; preRight[y - x - 1] = pre[y]; } }
問題2:實驗四樹的輸出格式不對,從而致使後綴表達式輸出格式不正確。
如圖
一樣是通過調試發現,在符號入棧的時候,轉化爲LinkedBinaryTree
類型存儲的時候,並以LinkedBinaryTree
類型爲根結點的時候,會根據其toString
方法,變成,\n\n + \n\n \n\n
,因此輸出樹的時候前面會多出不少行。
如圖
個人解決辦法是,把操做數棧改爲String
類型的,以String
類型的符號做爲根結點,這樣,結果能夠正常輸出。
如圖
問題3: 實驗四中綴轉後綴的括號問題。
// 處理括號 if (tempToken.equals("(")) { op.push(tempToken); tempToken = stringTokenizer.nextToken(); while (!tempToken.equals(")")) { if (tempToken.equals("+") || tempToken.equals("-")) { if (!op.isEmpty()) { // 棧不空,判斷「(」 if (op.peek().equals("(")) op.push(tempToken); else { String b = op.pop(); operand1 = getOperand(expre); operand2 = getOperand(expre); LinkedBinaryTree a = new LinkedBinaryTree(b, operand2, operand1); expre.push(a); op.push(tempToken); } } else { // 棧爲空,運算符入棧 op.push(tempToken); } } else if (tempToken.equals("*") || tempToken.equals("/")) { if (!op.isEmpty()) { if (op.peek().equals("*") || op.peek().equals("/")) { String b = op.pop(); operand1 = getOperand(expre); operand2 = getOperand(expre); LinkedBinaryTree a = new LinkedBinaryTree(b, operand2, operand1); expre.push(a); op.push(tempToken); } else { op.push(tempToken); } } } else { // 操做數入棧 LinkedBinaryTree a = new LinkedBinaryTree(tempToken); expre.push(a); } tempToken = stringTokenizer.nextToken(); } while (true) { String b = op.pop(); if (!b.equals("(")) { operand1 = getOperand(expre); operand2 = getOperand(expre); LinkedBinaryTree a = new LinkedBinaryTree(b, operand2, operand1); expre.push(a); } else { // 終止循環 break; } } }