欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語言判定一棵二叉樹是否為二叉搜索樹的方法分析

 更新時間:2018年08月29日 10:28:01   作者:xiaofei0859  
這篇文章主要介紹了C語言判定一棵二叉樹是否為二叉搜索樹的方法,結(jié)合實例形式綜合對比分析了C語言針對二叉搜索樹判定的原理、算法、效率及相關(guān)實現(xiàn)技巧,需要的朋友可以參考下

本文實例講述了C語言判定一棵二叉樹是否為二叉搜索樹的方法。分享給大家供大家參考,具體如下:

問題

給定一棵二叉樹,判定該二叉樹是否是二叉搜索樹(Binary Search Tree)?

解法1:暴力搜索

首先說明一下二叉樹和二叉搜索樹的區(qū)別。二叉樹指這樣的樹結(jié)構(gòu),它的每個結(jié)點的孩子數(shù)目最多為2個;二叉搜索樹是一種二叉樹,但是它有附加的一些約束條件,這些約束條件必須對每個結(jié)點都成立:

  • 結(jié)點node的左子樹所有結(jié)點的值都小于node的值。
  • 結(jié)點node的右子樹所有結(jié)點的值都大于node的值。
  • 結(jié)點node的左右子樹同樣都必須是二叉搜索樹。

該問題在面試中也許經(jīng)常問到,考察的是對二叉搜索樹定義的理解。初看這個問題,也許會想這樣來實現(xiàn):

假定當(dāng)前結(jié)點值為k。對于二叉樹中每個結(jié)點,判斷其左孩子的值是否小于k,其右孩子的值是否大于k。如果所有結(jié)點都滿足該條件,則該二叉樹是一棵二叉搜索樹。

很不幸的是,這個算法是錯誤的??紤]下面的二叉樹,它符合上面算法的條件,但是它不是一棵二叉搜索樹。

     10
   /  \
  5   15     -------- binary tree (1)
     /  \
    6    20

那么,根據(jù)二叉搜索樹的定義,可以想到一種暴力搜索的方法來判定二叉樹是否為二叉搜索樹。

假定當(dāng)前結(jié)點值為k。則對于二叉樹中每個結(jié)點,其左子樹所有結(jié)點的值必須都小于k,其右子樹所有結(jié)點的值都必須大于k。

暴力搜索算法代碼如下,雖然效率不高,但是它確實能夠完成工作。該解法最壞情況復(fù)雜度為O(n^2),n為結(jié)點數(shù)目。(當(dāng)所有結(jié)點都在一邊的時候出現(xiàn)最壞情況)

/*判斷左子樹的結(jié)點值是否都小于val*/
bool isSubTreeLessThan(BinaryTree *p, int val)
{
 if (!p) return true;
 return (p->data < val &&
     isSubTreeLessThan(p->left, val) &&
     isSubTreeLessThan(p->right, val));
}
/*判斷右子樹的結(jié)點值是否都大于val*/
bool isSubTreeGreaterThan(BinaryTree *p, int val)
{
 if (!p) return true;
 return (p->data > val &&
     isSubTreeGreaterThan(p->left, val) &&
     isSubTreeGreaterThan(p->right, val));
}
/*判定二叉樹是否是二叉搜索樹*/
bool isBSTBruteForce(BinaryTree *p)
{
 if (!p) return true;
 return isSubTreeLessThan(p->left, p->data) &&
     isSubTreeGreaterThan(p->right, p->data) &&
     isBSTBruteForce(p->left) &&
     isBSTBruteForce(p->right);
}

一個類似的解法是:對于結(jié)點node,判斷其左子樹最大值是否大于node的值,如果是,則該二叉樹不是二叉搜索樹。如果不是,則接著判斷右子樹最小值是否小于或等于node的值,如果是,則不是二叉搜索樹。如果不是則接著遞歸判斷左右子樹是否是二叉搜索樹。(代碼中的maxValue和minValue函數(shù)功能分別是返回二叉樹中的最大值和最小值,這里假定二叉樹為二叉搜索樹,實際返回的不一定是最大值和最小值)

int isBST(struct node* node)
{
 if (node==NULL) return(true);
 //如果左子樹最大值>=當(dāng)前node的值,則返回false
 if (node->left!=NULL && maxValue(node->left) >= node->data)
  return(false);
 // 如果右子樹最小值<=當(dāng)前node的值,返回false
 if (node->right!=NULL && minValue(node->right) <= node->data)
  return(false);
 // 如果左子樹或者右子樹不是BST,返回false
 if (!isBST(node->left) || !isBST(node->right))
  return(false);
 // 通過所有測試,返回true
 return(true);
}

解法2:更好的解法

以前面提到的binary tree(1)為例,當(dāng)我們從結(jié)點10遍歷到右結(jié)點15時,我們知道右子樹結(jié)點值肯定都在10和+INFINITY(無窮大)之間。當(dāng)我們遍歷到結(jié)點15的左孩子結(jié)點6時,我們知道結(jié)點15的左子樹結(jié)點值都必須在10到15之間。顯然,結(jié)點6不符合條件,因此它不是一棵二叉搜索樹。該算法代碼如下:

int isBST2(struct node* node)
{
   return(isBSTUtil(node, INT_MIN, INT_MAX));
}
/*
給定的二叉樹是BST則返回true,且它的值 >min 以及 < max.
*/
int isBSTUtil(struct node* node, int min, int max)
{
   if (node==NULL) return(true);
   // 如果不滿足min和max約束,返回false
   if (node->data<=min || node->data>=max) return(false);
   // 遞歸判斷左右子樹是否滿足min和max約束條件
   return
     isBSTUtil(node->left, min, node->data) &&
     isBSTUtil(node->right, node->data, max)
   );
}

由于該算法只需要訪問每個結(jié)點1次,因此時間復(fù)雜度為O(n),比解法1效率高很多。

解法3:中序遍歷算法

因為一棵二叉搜索樹的中序遍歷后其結(jié)點值是從小到大排好序的,所以依此給出下面的解法。該解法時間復(fù)雜度也是O(n)。

bool isBSTInOrder(BinaryTree *root)
{
 int prev = INT_MIN;
 return isBSTInOrderHelper(root, prev);
}
/*該函數(shù)判斷二叉樹p是否是一棵二叉搜索樹,且其結(jié)點值都大于prev*/
bool isBSTInOrderHelper(BinaryTree *p, int& prev)
{
 if (!p) return true;
 if (isBSTInOrderHelper(p->left, prev)) { // 如果左子樹是二叉搜索樹,且結(jié)點值都大于prev
  if (p->data > prev) { //判斷當(dāng)前結(jié)點值是否大于prev,因為此時prev已經(jīng)設(shè)置為已經(jīng)中序遍歷過的結(jié)點的最大值。
   prev = p->data;
   return isBSTInOrderHelper(p->right, prev); //若結(jié)點值大于prev,則設(shè)置prev為當(dāng)前結(jié)點值,并判斷右子樹是否二叉搜索樹且結(jié)點值都大于prev。
  } else {
   return false;
  }
 }
 else {
  return false;
 }
}

希望本文所述對大家C語言程序設(shè)計有所幫助。

相關(guān)文章

  • 用C語言求解第N項斐波那契數(shù)列問題

    用C語言求解第N項斐波那契數(shù)列問題

    這篇文章主要介紹了用C語言求解第N項斐波那契數(shù)列問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-11-11
  • C++實現(xiàn)bmp格式圖像讀寫

    C++實現(xiàn)bmp格式圖像讀寫

    這篇文章主要為大家詳細(xì)介紹了C++實現(xiàn)bmp格式圖像讀寫,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-10-10
  • C語言數(shù)組與地址、數(shù)組名到底是什么詳解

    C語言數(shù)組與地址、數(shù)組名到底是什么詳解

    在寫代碼的時候,我們經(jīng)常用到數(shù)組,那么有沒有想過數(shù)組名是什么呢?這篇文章主要給大家介紹了關(guān)于C語言數(shù)組與地址、數(shù)組名到底是什么的相關(guān)資料,文中通過實例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-06-06
  • C++對象內(nèi)存分布詳解(包括字節(jié)對齊和虛函數(shù)表)

    C++對象內(nèi)存分布詳解(包括字節(jié)對齊和虛函數(shù)表)

    下面小編就為大家?guī)硪黄狢++對象內(nèi)存分布詳解(包括字節(jié)對齊和虛函數(shù)表)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-12-12
  • C/C++ 獲取自身IP與域名片段的示例代碼

    C/C++ 獲取自身IP與域名片段的示例代碼

    這篇文章主要介紹了C/C++ 獲取自身IP與域名片段的示例代碼,幫助大家更好的理解和學(xué)習(xí)C/C++編程,感興趣的朋友可以了解下
    2020-10-10
  • C++ decltype類型說明符

    C++ decltype類型說明符

    在C++中,decltype作為操作符,用于查詢表達(dá)式的數(shù)據(jù)類型。decltype在C++11標(biāo)準(zhǔn)制定時引入,主要是為泛型編程而設(shè)計,以解決泛型編程中,由于有些類型由模板參數(shù)決定,而難以(甚至不可能)表示之的問題。
    2016-03-03
  • 用C語言實現(xiàn)從文本文件中讀取數(shù)據(jù)后進行排序的功能

    用C語言實現(xiàn)從文本文件中讀取數(shù)據(jù)后進行排序的功能

    這是一個十分可靠的程序,這個程序的查錯能力非常強悍。程序包含了文件操作,歸并排序和字符串輸入等多種技術(shù)。對大家學(xué)習(xí)C語言很有幫助,有需要的一起來看看。
    2016-08-08
  • 基于Matlab實現(xiàn)野狗優(yōu)化算法的示例代碼

    基于Matlab實現(xiàn)野狗優(yōu)化算法的示例代碼

    野狗優(yōu)化算法(Dingo?Optimization?Algorithm,?DOA)模仿澳大利亞野狗的社交行為。DOA算法的靈感來源于野狗的狩獵策略,即迫害攻擊、分組策略和食腐行為。本文將通過Matlab實現(xiàn)這一算法,感興趣的可以了解一下
    2022-04-04
  • C++ Qt屬性系統(tǒng)詳細(xì)介紹

    C++ Qt屬性系統(tǒng)詳細(xì)介紹

    這篇文章主要介紹了C++ Qt屬性系統(tǒng)詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下
    2016-12-12
  • C++ 使用CRC32檢測內(nèi)存映像完整性的實現(xiàn)步驟

    C++ 使用CRC32檢測內(nèi)存映像完整性的實現(xiàn)步驟

    當(dāng)我們使用動態(tài)補丁的時候,那么內(nèi)存中同樣不存在校驗效果,也就無法抵御對方動態(tài)修改機器碼了,為了防止解密者直接對內(nèi)存打補丁,我們需要在硬盤校驗的基礎(chǔ)上,增加內(nèi)存校驗,防止動態(tài)補丁的運用。
    2021-06-06

最新評論