最美情侣中文字幕电影,在线麻豆精品传媒,在线网站高清黄,久久黄色视频

歡迎光臨散文網(wǎng) 會員登陸 & 注冊

二叉搜索樹 Binary Search Tree 總結(jié)

2023-03-24 13:36 作者:Enderch  | 我要投稿

## 定義

二叉搜索樹是一種二叉樹的樹形數(shù)據(jù)結(jié)構(gòu)。定義如下:


- 空樹或左右子樹都是二叉搜索樹的二叉樹是二叉搜索樹。


- 二叉搜索樹的左子樹上的結(jié)點(diǎn)的權(quán)值都小于其根節(jié)點(diǎn)的權(quán)值。


- 二叉搜索樹的右子樹上的結(jié)點(diǎn)的權(quán)值都大于其根節(jié)點(diǎn)的權(quán)值。


而一個有 $n$ 個結(jié)點(diǎn)的二叉搜索樹的時間復(fù)雜度最優(yōu)情況下為 $O(\log n)$。但當(dāng)二叉搜索樹為鏈狀時時間復(fù)雜度最壞,為 $O(n)$。

## 存儲

我的寫法(其實(shí)是 wx 和鍬的)是用結(jié)構(gòu)體,分別存儲一個結(jié)點(diǎn)的父親、左右兒子和權(quán)值。

```c++

const int SON=2;

struct BST

{

int fa,ch[SON];//ch[0]為左兒子,ch[1]為右兒子。

int val;

}bst[MAXN];

```

## 詳細(xì)操作

### 1.遍歷

這里主要寫的是前序遍歷和中序遍歷。


前序非常簡單啊,根據(jù)定義就可以了。先輸出當(dāng)前結(jié)點(diǎn)的權(quán)值,再通過遞歸先遍歷左子樹,再遍歷右子樹。中序就是先左子樹,再輸出當(dāng)前結(jié)點(diǎn),再右子樹。

```c++

void Love_Theory(int x)

{//前序。

if(!x) return ;//遇到葉結(jié)點(diǎn)就返回。

printf(" %lld",bst[x].val);

Love_Theory(bst[x].ch[0]);

Love_Theory(bst[x].ch[1]);

}

void Love_Elegia(int x)

{//中序。

if(!x) return ;

Love_Elegia(bst[x].ch[0]);

printf(" %lld",bst[x].val);

Love_Elegia(bst[x].ch[1]);

}

```

### 2.查找

根據(jù)二叉搜索樹的定義,我們只需要先比較根節(jié)點(diǎn)與要查找的元素的值。如果小于根節(jié)點(diǎn),就跳到左子樹。大于就跳到右子樹,一直這樣下去直到找到這個元素。


然而查找的寫法肯定是先想到的遞歸,這里寫的是用 `while` 的方法。

```c++

int find(int v)

{

int x=root;

if(!x) return x;

bool temp=v>bst[x].val;//temp存儲要查找的元素是否大于當(dāng)前結(jié)點(diǎn)的權(quán)值。

while(v!=bst[x].val&&bst[x].ch[temp])

{

x=bst[x].ch[temp];//更新x找到下一個結(jié)點(diǎn)。

temp=v>bst[x].val;//更新temp。

}

return x;

}

```

### 3.插入

想要在二叉搜索樹中插入一個元素,首先應(yīng)該找到在哪插。這只需要用到前面查找時用的方法,一直找直到為空。而 `while` 循環(huán)會直到為空才會退出,所以在循環(huán)時需要另定義一個變量 $y$ 來更新當(dāng)前結(jié)點(diǎn) $x$ 的父親。


找到位置后,我們先需要初始化。將它的權(quán)值賦為插入的值,將它的父親、左右兒子初始化為 $0$,同時增加結(jié)點(diǎn)數(shù)。這個可以寫成一個函數(shù)。

```c++

int new_node(int v)

{

int x=++bst_cnt;

bst[x].val=v;

bst[x].fa=bst[x].ch[0]=bst[x].ch[1]=0;

return x;//這里返回的是新的結(jié)點(diǎn)數(shù)量,用于插入時更新

}

```

之后我們要分類討論。當(dāng)是空樹的情況時,故當(dāng) $y$ 等于 $0$ 時,這需要更新根節(jié)點(diǎn)為 $x$。另外就只需判斷 $x$ 與要插入的值的大小就行了,即判斷是插在左子樹還是右子樹。最后更新新結(jié)點(diǎn)的父親。


完整代碼:

```c++

void insert(int v)

{

int x=root,y=0;//從根節(jié)點(diǎn)開始找。

while(x&&bst[x].val!=v)

{

y=x;

bool temp=v>bst[x].val;

x=bst[x].ch[temp];

}

x=new_node(v);

if(!y) root=x;

else

{

bool temp=v>bst[y].val;

bst[y].ch[temp]=x;

}

bst[x].fa=y;

}

```

### 4.刪除

啊首先就是要分類討論,分別是刪除的結(jié)點(diǎn)沒有兒子、有一個兒子,有兩個兒子。


如果該結(jié)點(diǎn)沒有兒子,則將它的父親的子節(jié)點(diǎn)改為 $0$。


如果只有一個兒子,連接該結(jié)點(diǎn)的父親和它的子節(jié)點(diǎn)。


如果該結(jié)點(diǎn)有兩個兒子,找到該結(jié)點(diǎn)的后繼,刪除其后繼,將該結(jié)點(diǎn)的“遺產(chǎn)”都轉(zhuǎn)給它的后繼。


啊那要怎么找后繼呢。


一個結(jié)點(diǎn)的后繼的定義是在整棵樹中大于該結(jié)點(diǎn)的最小結(jié)點(diǎn),在結(jié)合二叉搜索樹的定義,我們可以發(fā)現(xiàn),要找一個結(jié)點(diǎn)的后繼,首先找到它的右兒子(右子樹上的所有結(jié)點(diǎn)一定大于它的根節(jié)點(diǎn)),再一直往左兒子方向遍歷,直到?jīng)]有左兒子(深度越大的左兒子越?。?。

```c++

int successor(int x)

{

x=bst[x].ch[1];//先往右。

while(bst[x].ch[0]) x=bst[x].ch[0];//然后一直往左跑,直到?jīng)]有左兒子。

return x;

}

```

啊這樣的話刪除操作的代碼我們也可以寫出來了。

```c++

int x=read();

x=find(x);

int y=bst[x].fa;

if((!bst[x].ch[0])&&(!bst[x].ch[1]))

{//沒有兒子。

bool temp=bst[x].val>bst[y].val;

bst[y].ch[temp]=0;//把它爹砍了。

}

else if(bst[x].ch[0]&&bst[x].ch[1])

{//有兩個兒子。

int z=successor(x);//找后繼

int z_fa=bst[z].fa;

bool temp=bst[z].val>bst[z_fa].val;

bst[z_fa].ch[temp]=bst[z].ch[1];

if(bst[z].ch[1]) bst[bst[z].ch[1]].fa=z_fa;//刪除后繼。

bst[z].fa=bst[x].fa;

? ? bst[z].ch[0]=bst[x].ch[0];

? ? bst[z].ch[1]=bst[x].ch[1];

? ??

? ? temp=bst[x].val>bst[y].val;

? ? if(y) bst[y].ch[temp]=z;

? ? if(bst[z].ch[0]) bst[bst[z].ch[0]].fa=z;

? ? if(bst[z].ch[1]) bst[bst[z].ch[1]].fa=z;//用后繼替代要刪除的結(jié)點(diǎn)。

}

else?

{//只有一個兒子。

int z;

? ? if(bst[x].ch[0]) z=bst[x].ch[0];

? ? else z=bst[x].ch[1];//判一下是有左兒子還是右兒子。

? ? bool temp=bst[x].val>bst[y].val;

? ? bst[y].ch[temp]=z;

? ? bst[z].fa=y;//連接它的父親與它的兒子(也就是直接跳過該結(jié)點(diǎn),不看它)。

}

```

## 總結(jié)

啊這樣的話二叉搜索樹的板子最終版就完成了?。?!

```c++

/*

ID: enderch1

PROG:

LANG: C++

*/

#include<bits/stdc++.h>

#define change_max(a,b) a=max(a,b)

#define change_min(a,b) a=min(a,b)

#define int long long

using namespace std;

const int MAXN=500005;

const int SON=2;

struct BST

{

int fa,ch[SON];

int val;

}bst[MAXN];

int root,bst_cnt;

int new_node(int v)

{

int x=++bst_cnt;

bst[x].val=v;

bst[x].fa=bst[x].ch[0]=bst[x].ch[1]=0;

return x;

}

void insert(int v)

{

int x=root,y=0;

while(x&&bst[x].val!=v)

{

y=x;

bool temp=v>bst[x].val;

x=bst[x].ch[temp];

}

x=new_node(v);

if(!y) root=x;

else

{

bool temp=v>bst[y].val;

bst[y].ch[temp]=x;

}

bst[x].fa=y;

}

int find(int v)

{

int x=root;

if(!x) return x;

bool temp=v>bst[x].val;

while(v!=bst[x].val&&bst[x].ch[temp])

{

x=bst[x].ch[temp];

temp=v>bst[x].val;

}

return x;

}

void Love_Theory(int x)

{

if(!x) return ;

printf(" %lld",bst[x].val);

Love_Theory(bst[x].ch[0]);

Love_Theory(bst[x].ch[1]);

}

void Love_Elegia(int x)

{

if(!x) return ;

Love_Elegia(bst[x].ch[0]);

printf(" %lld",bst[x].val);

Love_Elegia(bst[x].ch[1]);

}

int successor(int x)

{

x=bst[x].ch[1];

while(bst[x].ch[0]) x=bst[x].ch[0];

return x;

}

int read()

{

? ? int s=0,w=1;

? ? char ch=getchar();

? ? while(ch<'0'||ch>'9') {if(ch=='-') w=-1;ch=getchar();}

? ? while(ch>='0'&&ch<='9') {s=s*10+ch-'0';ch=getchar();}

? ? return s*w;

}

void solve()

{

int T=read();

while(T--)

{

string s;

cin>>s;

if(s=="insert")

{

int x=read();

insert(x);

}

else if(s=="find")

{

int x=read();

int temp=find(x);

if(x==bst[temp].val) puts("yes");

else puts("no");

}

else if(s=="delete")

{

int x=read();

x=find(x);

int y=bst[x].fa;

if((!bst[x].ch[0])&&(!bst[x].ch[1]))

{

bool temp=bst[x].val>bst[y].val;

bst[y].ch[temp]=0;

}

else if(bst[x].ch[0]&&bst[x].ch[1])

{

int z=successor(x);

int z_fa=bst[z].fa;

bool temp=bst[z].val>bst[z_fa].val;

bst[z_fa].ch[temp]=bst[z].ch[1];

if(bst[z].ch[1]) bst[bst[z].ch[1]].fa=z_fa;

? ? ? ? ? ? ? ? bst[z].fa=bst[x].fa;

? ? ? ? ? ? ? ? bst[z].ch[0]=bst[x].ch[0];

? ? ? ? ? ? ? ? bst[z].ch[1]=bst[x].ch[1];

? ? ? ? ? ? ? ??

? ? ? ? ? ? ? ? temp=bst[x].val>bst[y].val;

? ? ? ? ? ? ? ? if(y) bst[y].ch[temp]=z;

? ? ? ? ? ? ? ? if(bst[z].ch[0]) bst[bst[z].ch[0]].fa=z;

? ? ? ? ? ? ? ? if(bst[z].ch[1]) bst[bst[z].ch[1]].fa=z;

}

? ? ? ? ? ? else?

{

int z;

? ? ? ? ? ? ? ? if(bst[x].ch[0]) z=bst[x].ch[0];

? ? ? ? ? ? ? ? else z=bst[x].ch[1];

? ? ? ? ? ? ? ? bool temp=bst[x].val>bst[y].val;

? ? ? ? ? ? ? ? bst[y].ch[temp]=z;

? ? ? ? ? ? ? ? bst[z].fa=y;

? ? ? ? ? ? }

}

else

{

Love_Elegia(root);

putchar('\n');

Love_Theory(root);

putchar('\n');

}

}

}

signed main()

{

solve();

return 0;

}

```


啊看到這里你會發(fā)現(xiàn),我寫的 BST 巨長,而且呢,巨爛無比,啊還要防止退化成鏈。但是呢,親愛的 TQ 沒給我們講蹺蹺板樹。。。


等我學(xué)會了平衡樹就來補(bǔ)。。。


二叉搜索樹 Binary Search Tree 總結(jié)的評論 (共 條)

分享到微博請遵守國家法律
鄢陵县| 建瓯市| 铜山县| 辽中县| 伊金霍洛旗| 扶绥县| 泸西县| 阳山县| 黄平县| 沁阳市| 开平市| 城口县| 新沂市| 布尔津县| 三原县| 大埔县| 天等县| 南部县| 明溪县| 闻喜县| 揭阳市| 樟树市| 兰州市| 靖安县| 沂源县| 特克斯县| 桃园县| 马鞍山市| 高台县| 方正县| 彰化市| 泗阳县| 沐川县| 揭东县| 祁阳县| 招远市| 开原市| 义乌市| 揭东县| 古田县| 台山市|