Java實(shí)現(xiàn)Treap樹(shù)的示例代碼
Treap樹(shù)
Treap樹(shù)是平衡二叉搜索樹(shù)的一種實(shí)現(xiàn)方式,但它不是完全平衡的。平衡二叉搜索樹(shù)的實(shí)現(xiàn)方式還有AVL樹(shù)、紅黑樹(shù)、替罪羊樹(shù)、伸展樹(shù)
數(shù)據(jù)結(jié)構(gòu)
Treap樹(shù)的節(jié)點(diǎn)除了有二叉搜索樹(shù)的必須有的值,還有一個(gè)隨機(jī)生成的優(yōu)先級(jí)priority,供構(gòu)造小頂堆使用,小頂堆的特性就是父節(jié)點(diǎn)、左右子結(jié)點(diǎn)中永遠(yuǎn)是父節(jié)點(diǎn)的優(yōu)先級(jí)最小,最多和子結(jié)點(diǎn)的相等。而大頂堆則是父節(jié)點(diǎn)的最大。堆中左右子結(jié)點(diǎn)的優(yōu)先級(jí)并沒(méi)有特定要求
class TreeNode { ? ? int value; ? ? int priority; ? ? TreeNode left; ? ? TreeNode right; ? ? public TreeNode(int value, int priority) { ? ? ? ? this.value = value; ? ? ? ? this.priority = priority; ? ? } }
遍歷
Treap樹(shù)雖然是不完全平衡樹(shù),但是其完全滿(mǎn)足二叉搜索樹(shù)的特征,即中序遍歷得到的是有序數(shù)組
public void printTree(TreeNode root) { if (root != null) { printTree(root.left); System.out.println(root.value); printTree(root.right); } }
查詢(xún)
Treap樹(shù)滿(mǎn)足二叉搜索樹(shù)的特征,則直接根據(jù)其特征查詢(xún)
// 查詢(xún) // 根據(jù)二叉搜索樹(shù)性質(zhì)查詢(xún) public TreeNode query(TreeNode root, int value) { //這里的root才是真root,上面的方法只是局部變量 //所以不能在查詢(xún)中改變根節(jié)點(diǎn) TreeNode temp = root; while (temp != null) { if (temp.value > value) { temp = temp.left; } else if (temp.value < value) { temp = temp.right; } else { return temp; } } return null; }
增加
步驟
- 按照二叉搜索樹(shù)的插入方式,將節(jié)點(diǎn)插入到葉子節(jié)點(diǎn),如果在查找的過(guò)程找到要插入的值,則不會(huì)進(jìn)行插入,具有去重效果
- 插入節(jié)點(diǎn)后,根據(jù)隨機(jī)生成的priority優(yōu)先級(jí),按照小頂堆,即priority較小的成為父節(jié)點(diǎn),來(lái)進(jìn)行左旋右旋
//增加 // ?1.按照二叉查找樹(shù)的插入方式,將節(jié)點(diǎn)插入到樹(shù)葉中 // ?2.再根據(jù)priority優(yōu)先級(jí)的小頂堆性質(zhì)進(jìn)行左旋右旋 public TreeNode insert(int value, TreeNode root) { ? ? ?// 如果父節(jié)點(diǎn)為空,則創(chuàng)建一個(gè)父節(jié)點(diǎn)并返回 ? ? ?// 第一次父節(jié)點(diǎn)為根節(jié)點(diǎn) ? ? ?if (root == null) { ? ? ? ? ?return new TreeNode(value, random.nextInt()); ? ? ?} ? ? ?// 如果要根節(jié)點(diǎn)的值大于要插入結(jié)點(diǎn)的值,則應(yīng)該插入到根節(jié)點(diǎn)的左邊 ? ? ?if (root.value > value) { ? ? ? ? ?// 遞歸進(jìn)行插入,一直遞歸到葉子節(jié)點(diǎn)才會(huì)插入 ? ? ? ? ?// 如果遞歸到一個(gè)相等的節(jié)點(diǎn),則不會(huì)創(chuàng)建一個(gè)新節(jié)點(diǎn),會(huì)直接返回 ? ? ? ? ?root.left = insert(value, root.left); ? ? ? ? ?// 插入完成后,根據(jù)堆的優(yōu)先級(jí)進(jìn)行旋轉(zhuǎn)操作 ? ? ? ? ?// 左子結(jié)點(diǎn)的優(yōu)先級(jí)值小于根結(jié)點(diǎn)的優(yōu)先級(jí), ? ? ? ? ?// 根據(jù)小頂堆的規(guī)則,需要進(jìn)行右旋操作 ? ? ? ? ?if (root.left.priority < root.priority) { ? ? ? ? ? ? ?// 傳入root根結(jié)點(diǎn),返回的是左子結(jié)點(diǎn), ? ? ? ? ? ? ?// 但此時(shí)左子結(jié)點(diǎn)已經(jīng)旋轉(zhuǎn)成為根結(jié)點(diǎn)所以賦值給根結(jié)點(diǎn) ? ? ? ? ? ? ?root = rightRotate(root); ? ? ? ? ?} ? ? ?} ? ? ?// 如果根節(jié)點(diǎn)的值小于要插入結(jié)點(diǎn)的值大于,則應(yīng)該插入到根節(jié)點(diǎn)的右邊 ? ? ?else if (root.value < value) { ? ? ? ? ?// 遞歸插入 ? ? ? ? ?root.right = insert(value, root.right); ? ? ? ? ?// 右子結(jié)點(diǎn)的優(yōu)先級(jí)值小于根結(jié)點(diǎn)的優(yōu)先級(jí), ? ? ? ? ?// 根據(jù)小頂堆的規(guī)則,需要進(jìn)行左旋操作 ? ? ? ? ?if (root.right.priority < root.priority) { ? ? ? ? ? ? ?root = leftRotate(root); ? ? ? ? ?} ? ? ?} ? ? ?// 如果已經(jīng)有該值,則無(wú)須插入,什么都不動(dòng) ? ? ?else { ? ? ?} ? ? ?// 返回根結(jié)點(diǎn) ? ? ?return root; ?}
刪除
步驟
- 按照二叉搜索樹(shù)的特點(diǎn),先找到對(duì)應(yīng)的節(jié)點(diǎn)
- 若該結(jié)點(diǎn)為葉子結(jié)點(diǎn),則直接刪除,若該結(jié)點(diǎn)為非葉子節(jié)點(diǎn), 則進(jìn)行相應(yīng)的旋轉(zhuǎn),直到該結(jié)點(diǎn)為葉子節(jié)點(diǎn),然后進(jìn)行刪除
?//刪除、 // ? ? 1.根據(jù)二叉搜索樹(shù)的性質(zhì)找到相應(yīng)的結(jié)點(diǎn) // ? ? 2.若該結(jié)點(diǎn)為葉子結(jié)點(diǎn),則直接刪除,若該結(jié)點(diǎn)為非葉子節(jié)點(diǎn), // ? ? ? 則進(jìn)行相應(yīng)的旋轉(zhuǎn),直到該結(jié)點(diǎn)為葉子節(jié)點(diǎn),然后進(jìn)行刪除。 public TreeNode delete(int value, TreeNode root) { ? ? // 當(dāng)樹(shù)不為空才進(jìn)行刪除 ? ? if (root != null) { ? ? ? ? // 先進(jìn)行查找 ? ? ? ? // 往左找 ? ? ? ? if (root.value > value) { ? ? ? ? ? ? // 因?yàn)榭赡苷业搅撕髸?huì)進(jìn)行左旋右旋, ? ? ? ? ? ? // 所以其實(shí)左子結(jié)點(diǎn)會(huì)改變 ? ? ? ? ? ? root.left = delete(value, root.left); ? ? ? ? } ? ? ? ? // 往右找 ? ? ? ? else if (root.value < value) { ? ? ? ? ? ? root.right = delete(value, root.right); ? ? ? ? } ? ? ? ? //找到了 ? ? ? ? else { ? ? ? ? ? ? // 首先找到這里,root已經(jīng)變?yōu)槟繕?biāo)結(jié)點(diǎn),如果root是葉子結(jié)點(diǎn)刪去即可 ? ? ? ? ? ? if (root.left == null && root.right == null) { ? ? ? ? ? ? ? ? // 返回當(dāng)前節(jié)點(diǎn),即刪除后的樣子 null ? ? ? ? ? ? ? ? // 會(huì)遞歸到其父節(jié)點(diǎn)的root.right = delete(value, root.right); ? ? ? ? ? ? ? ? // 即指向了root.right = null 或 root.left = null,即刪除了我們要?jiǎng)h除的節(jié)點(diǎn) ? ? ? ? ? ? ? ? return null; ? ? ? ? ? ? } else if (root.left != null && root.right != null) { ? ? ? ? ? ? ? ? // 如果root左右子結(jié)點(diǎn)健在 ? ? ? ? ? ? ? ? // 此時(shí)就是想把目標(biāo)結(jié)點(diǎn)旋轉(zhuǎn)到底層去, ? ? ? ? ? ? ? ? // 然后需要選擇一個(gè)優(yōu)先級(jí)值比較小的結(jié)點(diǎn)放在目標(biāo)結(jié)點(diǎn)位置 ? ? ? ? ? ? ? ? // 如果左子結(jié)點(diǎn)優(yōu)先級(jí)較小 ? ? ? ? ? ? ? ? if (root.left.priority < root.right. priority) { ? ? ? ? ? ? ? ? ? ? // 旋轉(zhuǎn)root已經(jīng)變?yōu)樽笞咏Y(jié)點(diǎn),原來(lái)的根結(jié)點(diǎn)變?yōu)橛易庸?jié)點(diǎn) ? ? ? ? ? ? ? ? ? ? root = rightRotate(root); ? ? ? ? ? ? ? ? ? ? // 去找那被換下去的節(jié)點(diǎn),將它刪除掉 ? ? ? ? ? ? ? ? ? ? root.right = delete(value, root.right); ? ? ? ? ? ? ? ? } else { ? ? ? ? ? ? ? ? ? ? root = leftRotate(root); ? ? ? ? ? ? ? ? ? ? // 去找那被換下去的節(jié)點(diǎn),將它刪除掉 ? ? ? ? ? ? ? ? ? ? root.left = delete(value, root.left); ? ? ? ? ? ? ? ? } ? ? ? ? ? ? } ? ? ? ? ? ?else if (root.left != null) { ? ? ? ? ? ? ? ? // 沒(méi)有右子節(jié)點(diǎn),只能右旋了 ? ? ? ? ? ? ? ? root = rightRotate(root); ? ? ? ? ? ? ? ? // 去找那被換下去的節(jié)點(diǎn),將它刪除掉 ? ? ? ? ? ? ? ? root.right = delete(value, root.right); ? ? ? ? ? ? } ? ? ? ? ? ? else if (root.right != null) { ? ? ? ? ? ? ? ? // 沒(méi)有左子節(jié)點(diǎn),只能左旋了 ? ? ? ? ? ? ? ? root = leftRotate(root); ? ? ? ? ? ? ? ? // 去找那被換下去的節(jié)點(diǎn),將它刪除掉 ? ? ? ? ? ? ? ? root.left = delete(value, root.left); ? ? ? ? ? ? } ? ? ? ? } ? ? } ? ? return root; }
完整代碼
public class Treap { ? ? // 優(yōu)先級(jí)隨機(jī)數(shù)發(fā)生器 ? ? private static final Random random = new Random(); ? ? //增加 // ?1.按照二叉查找樹(shù)的插入方式,將節(jié)點(diǎn)插入到樹(shù)葉中 // ?2.再根據(jù)priority優(yōu)先級(jí)的小頂堆性質(zhì)進(jìn)行左旋右旋 ? ? public TreeNode insert(int value, TreeNode root) { ? ? ? ? // 如果父節(jié)點(diǎn)為空,則創(chuàng)建一個(gè)父節(jié)點(diǎn)并返回 ? ? ? ? // 第一次父節(jié)點(diǎn)為根節(jié)點(diǎn) ? ? ? ? if (root == null) { ? ? ? ? ? ? return new TreeNode(value, random.nextInt()); ? ? ? ? } ? ? ? ? // 如果要根節(jié)點(diǎn)的值大于要插入結(jié)點(diǎn)的值,則應(yīng)該插入到根節(jié)點(diǎn)的左邊 ? ? ? ? if (root.value > value) { ? ? ? ? ? ? // 遞歸進(jìn)行插入,一直遞歸到葉子節(jié)點(diǎn)才會(huì)插入 ? ? ? ? ? ? // 如果遞歸到一個(gè)相等的節(jié)點(diǎn),則不會(huì)創(chuàng)建一個(gè)新節(jié)點(diǎn),會(huì)直接返回 ? ? ? ? ? ? root.left = insert(value, root.left); ? ? ? ? ? ? // 插入完成后,根據(jù)堆的優(yōu)先級(jí)進(jìn)行旋轉(zhuǎn)操作 ? ? ? ? ? ? // 左子結(jié)點(diǎn)的優(yōu)先級(jí)值小于根結(jié)點(diǎn)的優(yōu)先級(jí), ? ? ? ? ? ? // 根據(jù)小頂堆的規(guī)則,需要進(jìn)行右旋操作 ? ? ? ? ? ? if (root.left.priority < root.priority) { ? ? ? ? ? ? ? ? // 傳入root根結(jié)點(diǎn),返回的是左子結(jié)點(diǎn), ? ? ? ? ? ? ? ? // 但此時(shí)左子結(jié)點(diǎn)已經(jīng)旋轉(zhuǎn)成為根結(jié)點(diǎn)所以賦值給根結(jié)點(diǎn) ? ? ? ? ? ? ? ? root = rightRotate(root); ? ? ? ? ? ? } ? ? ? ? } ? ? ? ? // 如果根節(jié)點(diǎn)的值小于要插入結(jié)點(diǎn)的值大于,則應(yīng)該插入到根節(jié)點(diǎn)的右邊 ? ? ? ? else if (root.value < value) { ? ? ? ? ? ? // 遞歸插入 ? ? ? ? ? ? root.right = insert(value, root.right); ? ? ? ? ? ? // 右子結(jié)點(diǎn)的優(yōu)先級(jí)值小于根結(jié)點(diǎn)的優(yōu)先級(jí), ? ? ? ? ? ? // 根據(jù)小頂堆的規(guī)則,需要進(jìn)行左旋操作 ? ? ? ? ? ? if (root.right.priority < root.priority) { ? ? ? ? ? ? ? ? root = leftRotate(root); ? ? ? ? ? ? } ? ? ? ? } ? ? ? ? // 如果已經(jīng)有該值,則無(wú)須插入,什么都不動(dòng) ? ? ? ? else { ? ? ? ? } ? ? ? ? // 返回根結(jié)點(diǎn) ? ? ? ? return root; ? ? } ? ? //刪除、 // ? ? 1.根據(jù)二叉搜索樹(shù)的性質(zhì)找到相應(yīng)的結(jié)點(diǎn) // ? ? 2.若該結(jié)點(diǎn)為葉子結(jié)點(diǎn),則直接刪除,若該結(jié)點(diǎn)為非葉子節(jié)點(diǎn), // ? ? ? 則進(jìn)行相應(yīng)的旋轉(zhuǎn),直到該結(jié)點(diǎn)為葉子節(jié)點(diǎn),然后進(jìn)行刪除。 ? ? public TreeNode delete(int value, TreeNode root) { ? ? ? ? // 當(dāng)樹(shù)不為空才進(jìn)行刪除 ? ? ? ? if (root != null) { ? ? ? ? ? ? // 先進(jìn)行查找 ? ? ? ? ? ? // 往左找 ? ? ? ? ? ? if (root.value > value) { ? ? ? ? ? ? ? ? // 因?yàn)榭赡苷业搅撕髸?huì)進(jìn)行左旋右旋, ? ? ? ? ? ? ? ? // 所以其實(shí)左子結(jié)點(diǎn)會(huì)改變 ? ? ? ? ? ? ? ? root.left = delete(value, root.left); ? ? ? ? ? ? } ? ? ? ? ? ? // 往右找 ? ? ? ? ? ? else if (root.value < value) { ? ? ? ? ? ? ? ? root.right = delete(value, root.right); ? ? ? ? ? ? } ? ? ? ? ? ? //找到了 ? ? ? ? ? ? else { ? ? ? ? ? ? ? ? // 首先找到這里,root已經(jīng)變?yōu)槟繕?biāo)結(jié)點(diǎn),如果root是葉子結(jié)點(diǎn)刪去即可 ? ? ? ? ? ? ? ? if (root.left == null && root.right == null) { ? ? ? ? ? ? ? ? ? ? // 返回當(dāng)前節(jié)點(diǎn),即刪除后的樣子 null ? ? ? ? ? ? ? ? ? ? // 會(huì)遞歸到其父節(jié)點(diǎn)的root.right = delete(value, root.right); ? ? ? ? ? ? ? ? ? ? // 即指向了root.right = null 或 root.left = null,即刪除了我們要?jiǎng)h除的節(jié)點(diǎn) ? ? ? ? ? ? ? ? ? ? return null; ? ? ? ? ? ? ? ? } else if (root.left != null && root.right != null) { ? ? ? ? ? ? ? ? ? ? // 如果root左右子結(jié)點(diǎn)健在 ? ? ? ? ? ? ? ? ? ? // 此時(shí)就是想把目標(biāo)結(jié)點(diǎn)旋轉(zhuǎn)到底層去, ? ? ? ? ? ? ? ? ? ? // 然后需要選擇一個(gè)優(yōu)先級(jí)值比較小的結(jié)點(diǎn)放在目標(biāo)結(jié)點(diǎn)位置 ? ? ? ? ? ? ? ? ? ? // 如果左子結(jié)點(diǎn)優(yōu)先級(jí)較小 ? ? ? ? ? ? ? ? ? ? if (root.left.priority < root.right. priority) { ? ? ? ? ? ? ? ? ? ? ? ? // 旋轉(zhuǎn)root已經(jīng)變?yōu)樽笞咏Y(jié)點(diǎn),原來(lái)的根結(jié)點(diǎn)變?yōu)橛易庸?jié)點(diǎn) ? ? ? ? ? ? ? ? ? ? ? ? root = rightRotate(root); ? ? ? ? ? ? ? ? ? ? ? ? // 去找那被換下去的節(jié)點(diǎn),將它刪除掉 ? ? ? ? ? ? ? ? ? ? ? ? root.right = delete(value, root.right); ? ? ? ? ? ? ? ? ? ? } else { ? ? ? ? ? ? ? ? ? ? ? ? root = leftRotate(root); ? ? ? ? ? ? ? ? ? ? ? ? // 去找那被換下去的節(jié)點(diǎn),將它刪除掉 ? ? ? ? ? ? ? ? ? ? ? ? root.left = delete(value, root.left); ? ? ? ? ? ? ? ? ? ? } ? ? ? ? ? ? ? ? } ? ? ? ? ? ? ? ?else if (root.left != null) { ? ? ? ? ? ? ? ? ? ? // 沒(méi)有右子節(jié)點(diǎn),只能右旋了 ? ? ? ? ? ? ? ? ? ? root = rightRotate(root); ? ? ? ? ? ? ? ? ? ? // 去找那被換下去的節(jié)點(diǎn),將它刪除掉 ? ? ? ? ? ? ? ? ? ? root.right = delete(value, root.right); ? ? ? ? ? ? ? ? } ? ? ? ? ? ? ? ? else if (root.right != null) { ? ? ? ? ? ? ? ? ? ? // 沒(méi)有左子節(jié)點(diǎn),只能左旋了 ? ? ? ? ? ? ? ? ? ? root = leftRotate(root); ? ? ? ? ? ? ? ? ? ? // 去找那被換下去的節(jié)點(diǎn),將它刪除掉 ? ? ? ? ? ? ? ? ? ? root.left = delete(value, root.left); ? ? ? ? ? ? ? ? } ? ? ? ? ? ? } ? ? ? ? } ? ? ? ? return root; ? ? } ? ? // 查詢(xún) ? ? // 根據(jù)二叉搜索樹(shù)性質(zhì)查詢(xún) ? ? public TreeNode query(TreeNode root, int value) { ? ? ? ? //這里的root才是真root,上面的方法只是局部變量 ? ? ? ? //所以不能在查詢(xún)中改變根節(jié)點(diǎn) ? ? ? ? TreeNode temp = root; ? ? ? ? while (temp != null) { ? ? ? ? ? ? if (temp.value > value) { ? ? ? ? ? ? ? ? temp = temp.left; ? ? ? ? ? ? } else if (temp.value < value) { ? ? ? ? ? ? ? ? temp = temp.right; ? ? ? ? ? ? } else { ? ? ? ? ? ? ? ? return temp; ? ? ? ? ? ? } ? ? ? ? } ? ? ? ? return null; ? ? } ? ? // 右旋,左子節(jié)點(diǎn)右旋 ? ? public TreeNode rightRotate(TreeNode treeNode) { ? ? ? ? // temp為左子結(jié)點(diǎn) ? ? ? ? TreeNode temp = treeNode.left; ? ? ? ? //將父結(jié)點(diǎn)的左邊指向 temp的右子結(jié)點(diǎn) ? ? ? ? treeNode.left = temp.right; ? ? ? ? // 將temp結(jié)點(diǎn)的右邊指向父結(jié)點(diǎn) ? ? ? ? temp.right = treeNode; ? ? ? ? // 進(jìn)行上面兩步操作,在紙上畫(huà)一下就找到其右旋成功了, ? ? ? ? // 即左子結(jié)點(diǎn)變?yōu)楦Y(jié)點(diǎn)了 ? ? ? ? // 返回此時(shí)旋轉(zhuǎn)后的真正根結(jié)點(diǎn) ? ? ? ? return temp; ? ? } ? ? // 左旋,右子結(jié)點(diǎn)左旋 ? ? public TreeNode leftRotate(TreeNode treeNode) { ? ? ? ? // temp為右子結(jié)點(diǎn) ? ? ? ? TreeNode temp = treeNode.right; ? ? ? ? //將父結(jié)點(diǎn)的右邊指向 temp的左子結(jié)點(diǎn) ? ? ? ? treeNode.right = temp.left; ? ? ? ? // 將temp結(jié)點(diǎn)的左邊指向父結(jié)點(diǎn) ? ? ? ? temp.left = treeNode; ? ? ? ? // 進(jìn)行上面兩步操作,在紙上畫(huà)一下就找到其左旋成功了, ? ? ? ? // 即右子結(jié)點(diǎn)變?yōu)楦Y(jié)點(diǎn)了 ? ? ? ? // 返回此時(shí)旋轉(zhuǎn)后的真正根結(jié)點(diǎn) ? ? ? ? return temp; ? ? } ? ? public void printTree(TreeNode root) { ? ? ? ? if (root != null) { ? ? ? ? ? ? printTree(root.left); ? ? ? ? ? ? System.out.println(root.value); ? ? ? ? ? ? printTree(root.right); ? ? ? ? } ? ? } ? ? public static void main(String[] args) { ? ? ? ? Treap treap = new Treap(); ? ? ? ? TreeNode root = null; ? ? ? ? root = treap.insert(1, root); ? ? ? ? root = treap.insert(2, root); ? ? ? ? root = treap.insert(3, root); ? ? ? ? root = treap.insert(4, root); ? ? ? ? root = treap.insert(5, root); ? ? ? ? root = treap.insert(6, root); ? ? ? ? //中序遍歷,如果打印的值由小到大,說(shuō)明滿(mǎn)足二叉搜索樹(shù)特征 ? ? ? ? treap.printTree(root); ? ? ? ? System.out.println(); ? ? ? ? // 測(cè)試查詢(xún) ? ? ? ? TreeNode query = treap.query(root, 1); ? ? ? ? System.out.println(query.value); ? ? ? ? query = treap.query(root, 2); ? ? ? ? System.out.println(query.value); ? ? ? ? query = treap.query(root, 3); ? ? ? ? System.out.println(query.value); ? ? ? ? query = treap.query(root, 4); ? ? ? ? System.out.println(query.value); ? ? ? ? query = treap.query(root, 5); ? ? ? ? System.out.println(query.value); ? ? ? ? query = treap.query(root, 6); ? ? ? ? System.out.println(query.value); ? ? ? ? query = treap.query(root, 7); ? ? ? ? System.out.println(query); ? ? ? ? System.out.println(); ? ? ? ? // 測(cè)試刪除 ? ? ? ? root = treap.delete(2,root); ? ? ? ? root = treap.delete(3,root); ? ? ? ? root = treap.delete(5,root); ? ? ? ? root = treap.delete(7,root); ? ? ? ? treap.printTree(root); ? ? } } class TreeNode { ? ? int value; ? ? int priority; ? ? TreeNode left; ? ? TreeNode right; ? ? public TreeNode(int value, int priority) { ? ? ? ? this.value = value; ? ? ? ? this.priority = priority; ? ? } }
到此這篇關(guān)于Java實(shí)現(xiàn)Treap樹(shù)的示例代碼的文章就介紹到這了,更多相關(guān)Java Treap樹(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java創(chuàng)建二叉搜索樹(shù),實(shí)現(xiàn)搜索,插入,刪除的操作實(shí)例
- Java 實(shí)現(xiàn)二叉搜索樹(shù)的查找、插入、刪除、遍歷
- java實(shí)現(xiàn) 二叉搜索樹(shù)功能
- Java刪除二叉搜索樹(shù)的任意元素的方法詳解
- Java二叉搜索樹(shù)基礎(chǔ)原理與實(shí)現(xiàn)方法詳解
- Java二叉搜索樹(shù)遍歷操作詳解【前序、中序、后序、層次、廣度優(yōu)先遍歷】
- Java刪除二叉搜索樹(shù)最大元素和最小元素的方法詳解
- 劍指Offer之Java算法習(xí)題精講字符串與二叉搜索樹(shù)
- 劍指Offer之Java算法習(xí)題精講字符串操作與數(shù)組及二叉搜索樹(shù)
相關(guān)文章
mybatis foreach遍歷LIST讀到數(shù)據(jù)為null的問(wèn)題
這篇文章主要介紹了mybatis foreach遍歷LIST讀到數(shù)據(jù)為null的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。2022-02-02詳解SpringMVC使用MultipartFile實(shí)現(xiàn)文件的上傳
本篇文章主要介紹了SpringMVC使用MultipartFile實(shí)現(xiàn)文件的上傳,本地的文件上傳到資源服務(wù)器上,比較好的辦法就是通過(guò)ftp上傳。這里是結(jié)合SpringMVC+ftp的形式上傳的,有興趣的可以了解一下。2016-12-12解決mybatis批量更新出現(xiàn)SQL報(bào)錯(cuò)問(wèn)題
這篇文章主要介紹了mybatis批量更新出現(xiàn)SQL報(bào)錯(cuò),解決辦法也很簡(jiǎn)單只需要在application.properties配置文中的數(shù)據(jù)源url后面添加一個(gè)參數(shù),需要的朋友可以參考下2022-02-02Spring中@Transactional用法詳細(xì)介紹
這篇文章主要介紹了Spring中@Transactional用法詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下2017-02-02Java使用POI實(shí)現(xiàn)excel文件的導(dǎo)入和導(dǎo)出
這篇文章主要為大家詳細(xì)介紹了Java如何使用POI實(shí)現(xiàn)excel文件的導(dǎo)入和導(dǎo)出功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-12-12迅速學(xué)會(huì)@ConfigurationProperties的使用操作
這篇文章主要介紹了迅速學(xué)會(huì)@ConfigurationProperties的使用,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10SpringBoot2.x 整合 AntiSamy防御XSS攻擊的簡(jiǎn)單總結(jié)
本文主要對(duì)SpringBoot2.x集成AntiSamy防御XSS攻擊進(jìn)行簡(jiǎn)單總結(jié),其中SpringBoot使用的2.4.5版本,通過(guò)示例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧2021-08-08