二叉查找樹(shù)的插入,刪除,查找
二叉查找樹(shù)是滿足以下條件的二叉樹(shù):
1、左子樹(shù)上的所有節(jié)點(diǎn)值均小于根節(jié)點(diǎn)值,
2、右子樹(shù)上的所有節(jié)點(diǎn)值均不小于根節(jié)點(diǎn)值,
3、左右子樹(shù)也滿足上述兩個(gè)條件。
二叉查找樹(shù)的插入過(guò)程如下:
1.若當(dāng)前的二叉查找樹(shù)為空,則插入的元素為根節(jié)點(diǎn),
2.若插入的元素值小于根節(jié)點(diǎn)值,則將元素插入到左子樹(shù)中,
3.若插入的元素值不小于根節(jié)點(diǎn)值,則將元素插入到右子樹(shù)中。
二叉查找樹(shù)的刪除,分三種情況進(jìn)行處理:
1.p為葉子節(jié)點(diǎn),直接刪除該節(jié)點(diǎn),再修改其父節(jié)點(diǎn)的指針(注意分是根節(jié)點(diǎn)和不是根節(jié)點(diǎn)),如圖a。
2.p為單支節(jié)點(diǎn)(即只有左子樹(shù)或右子樹(shù))。讓p的子樹(shù)與p的父親節(jié)點(diǎn)相連,刪除p即可;(注意分是根節(jié)點(diǎn)和不是根節(jié)點(diǎn));如圖b。
3.p的左子樹(shù)和右子樹(shù)均不空。找到p的后繼y,因?yàn)閥一定沒(méi)有左子樹(shù),所以可以刪除y,并讓y的父親節(jié)點(diǎn)成為y的右子樹(shù)的父親節(jié)點(diǎn),并用y的值代替p的值;或者方法二是找到p的前驅(qū)x,x一定沒(méi)有右子樹(shù),所以可以刪除x,并讓x的父親節(jié)點(diǎn)成為y的左子樹(shù)的父親節(jié)點(diǎn)。如圖c。
插入節(jié)點(diǎn)的代碼:
struct node
{
int val;
pnode lchild;
pnode rchild;
};
pnode BT = NULL;
//遞歸方法插入節(jié)點(diǎn)
pnode insert(pnode root, int x)
{
pnode p = (pnode)malloc(LEN);
p->val = x;
p->lchild = NULL;
p->rchild = NULL;
if(root == NULL){
root = p;
}
else if(x < root->val){
root->lchild = insert(root->lchild, x);
}
else{
root->rchild = insert(root->rchild, x);
}
return root;
}
//非遞歸方法插入節(jié)點(diǎn)
void insert_BST(pnode q, int x)
{
pnode p = (pnode)malloc(LEN);
p->val = x;
p->lchild = NULL;
p->rchild = NULL;
if(q == NULL){
BT = p;
return ;
}
while(q->lchild != p && q->rchild != p){
if(x < q->val){
if(q->lchild){
q = q->lchild;
}
else{
q->lchild = p;
}
}
else{
if(q->rchild){
q = q->rchild;
}
else{
q->rchild = p;
}
}
}
return;
}
查找節(jié)點(diǎn)的代碼:
pnode search_BST(pnode p, int x)
{
bool solve = false;
while(p && !solve){
if(x == p->val){
solve = true;
}
else if(x < p->val){
p = p->lchild;
}
else{
p = p->rchild;
}
}
if(p == NULL){
cout << "沒(méi)有找到" << x << endl;
}
return p;
}
刪除節(jié)點(diǎn)的代碼
bool delete_BST(pnode p, int x) //返回一個(gè)標(biāo)志,表示是否找到被刪元素
{
bool find = false;
pnode q;
p = BT;
while(p && !find){ //尋找被刪元素
if(x == p->val){ //找到被刪元素
find = true;
}
else if(x < p->val){ //沿左子樹(shù)找
q = p;
p = p->lchild;
}
else{ //沿右子樹(shù)找
q = p;
p = p->rchild;
}
}
if(p == NULL){ //沒(méi)找到
cout << "沒(méi)有找到" << x << endl;
}
if(p->lchild == NULL && p->rchild == NULL){ //p為葉子節(jié)點(diǎn)
if(p == BT){ //p為根節(jié)點(diǎn)
BT = NULL;
}
else if(q->lchild == p){
q->lchild = NULL;
}
else{
q->rchild = NULL;
}
free(p); //釋放節(jié)點(diǎn)p
}
else if(p->lchild == NULL || p->rchild == NULL){ //p為單支子樹(shù)
if(p == BT){ //p為根節(jié)點(diǎn)
if(p->lchild == NULL){
BT = p->rchild;
}
else{
BT = p->lchild;
}
}
else{
if(q->lchild == p && p->lchild){ //p是q的左子樹(shù)且p有左子樹(shù)
q->lchild = p->lchild; //將p的左子樹(shù)鏈接到q的左指針上
}
else if(q->lchild == p && p->rchild){
q->lchild = p->rchild;
}
else if(q->rchild == p && p->lchild){
q->rchild = p->lchild;
}
else{
q->rchild = p->rchild;
}
}
free(p);
}
else{ //p的左右子樹(shù)均不為空
pnode t = p;
pnode s = p->lchild; //從p的左子節(jié)點(diǎn)開(kāi)始
while(s->rchild){ //找到p的前驅(qū),即p左子樹(shù)中值最大的節(jié)點(diǎn)
t = s;
s = s->rchild;
}
p->val = s->val; //把節(jié)點(diǎn)s的值賦給p
if(t == p){
p->lchild = s->lchild;
}
else{
t->rchild = s->lchild;
}
free(s);
}
return find;
}
上一篇:淺析內(nèi)存對(duì)齊與ANSI C中struct型數(shù)據(jù)的內(nèi)存布局
欄 目:C語(yǔ)言
下一篇:c語(yǔ)言中if 語(yǔ)句的作用范圍示例代碼
本文標(biāo)題:二叉查找樹(shù)的插入,刪除,查找
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/4181.html
您可能感興趣的文章
- 01-10深入二叉樹(shù)兩個(gè)結(jié)點(diǎn)的最低共同父結(jié)點(diǎn)的詳解
- 01-10深入理解二叉樹(shù)的非遞歸遍歷
- 01-10深入遍歷二叉樹(shù)的各種操作詳解(非遞歸遍歷)
- 01-10linux c 查找使用庫(kù)的cflags與libs的方法詳解
- 01-10判斷整數(shù)序列是否為二元查找樹(shù)的后序遍歷結(jié)果的解決方法
- 01-10探討:C++實(shí)現(xiàn)鏈?zhǔn)蕉鏄?shù)(用非遞歸方式先序,中序,后序遍歷二叉樹(shù)
- 01-10如何在二叉樹(shù)中找出和為某一值的所有路徑
- 01-10深入linux下遍歷目錄樹(shù)的方法總結(jié)分析
- 01-10先序遍歷二叉樹(shù)的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)深入解析
- 01-10二叉搜索樹(shù)的插入與刪除(詳細(xì)解析)


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 04-02jquery與jsp,用jquery
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10C#中split用法實(shí)例總結(jié)
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10delphi制作wav文件的方法