Tree 小结

以下是数据结构部分的主要知识点的思维导图

tree_introduce.png

在这段时间基本上刷的都是跟二叉树有关的题目,所以下面主要针对二叉树部分进行总结。
  二叉树的各种操作基本上离不开基本的几个点,遍历、搜索、节点插入和删除。利用这几个操作互相结合就能解决大部分问题,当然前提是先理解它们的意义和熟悉操作代码的编写。下面分别对这些操作进行代码的实现与分析。

二叉树操作.png

假设树的结构体实现如下:

 Definition for a binary tree node.
  struct TreeNode {
      int val;
      TreeNode *left;
      TreeNode *right;
      TreeNode(int x) : val(x), left(NULL), right(NULL) {}
  };

遍历 (前序、中序、后序)

以中序遍历为例(其他两种情况大同小异),假设现在需要对一个根节点为root(不为空)的树进行遍历将节点的值按顺序放到vector中,这种情况下同常有递归和非递归两种实现方式,实现代码如下:

  • 递归实现
class Solution {
public:
    vector<int> traverse(TreeNode* root, vector<int>& vec) {
         traverse(root->left);
         vec.push_back(root->val);
         traverse(root->right);
         return vec;
   }
};
  • 使用栈实现h

1.遍历后树会发生变化的实现方式

class Solution {
public:
    vector<int> traverse(TreeNode* root, vector<int>& vec) {
           stack<TreeNode*> s;
           s.push(root);
           while(!s.empty()){
             TreeNode* node = s.top();
              if(node ->left){
                  s.push(node ->left);
                  node ->left = NULL;//防止重复遍历
              }else{
                  s.pop();
                  vec->push(node->val);
                   if(node->right){
                     s.push(node->right);
                   }   
              } 
          }
       }
       return vec;
   }
};

2.遍历后原树不会发生变化(使用unordered_map)

class Solution {
public:
    vector<int> traverse(TreeNode* root, vector<int>& vec) {
           unordered_map<TreeNode*, bool> map;
           stack<TreeNode*> s;
           s.push(root);
           while(!s.empty()){
             TreeNode* node = s.top();
              if(node ->left && !map[node]){
                  s.push(node ->left);
                  map[node->left] = true;
              }else{
                  s.pop();
                  vec->push(node->val);
                   if(node->right){
                     s.push(node->right);
                   }   
              } 
          }
       }
       return vec;
   }
};

3.遍历后原树不会发生变化(不使用unordered_map)

class Solution {
public:
    vector<int> inorderTraversal(TreeNode *root) {
        vector<int> vector;
        stack<TreeNode *> stack;
        TreeNode *pCurrent = root;

        while(!stack.empty() || pCurrent)
        {
            if(pCurrent)
            {
                //同一根节点下的左右节点,左节点比右节点先出栈(即左节点后进栈,其实最后所有进栈的节点都可以视左一个树中的中间节点。)
                stack.push(pCurrent);
                pCurrent = pCurrent->left;//若左节点为空则,该节点可以等同为中间节点,相对于其右节点先出栈(后进栈)
            }//节点为空就出栈
            else
            {//当左子节点或右子节点没有左子节点时 改节点出栈
                TreeNode *pNode = stack.top();
                vector.push_back(pNode->val);
                stack.pop();
                pCurrent = pNode->right;
            }
        }
        return vector;
    }
};

由于篇幅所限前序遍历和后序遍历的代码就不贴了,可去这个地方查看:
https://github.com/kid1943/leetcode/tree/master/tree/Binary%20Tree%20Preorder%20Traversal

https://github.com/kid1943/leetcode/tree/master/tree/Binary%20Tree%20Postorder%20Traversal

逐层遍历

  • 自上而下
class Solution {
public:
    vector<int> traverse(TreeNode* root) {
        queue<TreeNode*> q;
        queue<int> level;
        
        q.push(root);
        level.push(0);
        
        vector<int> vec;
        int m = -1;
        int result;
        while(q.size()){
          TreeNode* sr = q.front();
          result = sr->val;
          int l = level.front();
          level.pop();
          q.pop();  
          
          if(sr->left){
              q.push(sr->left);
              level.push(l+1);
          }
          
          if(sr->right){
              q.push(sr->right);
              level.push(l+1);
          }
        }
        return level;
    }
};

节点的删除

要删除二叉搜索树中的某个节点p需要考虑三种情况:
1)p有两颗非空子树
2)p是树叶
3)p是只有一颗非空子树

删除p节点的思路
1)要删除的节点p具有两颗非空子树。先将该节点的元素替换为它的左子树的最大元素或右子树的最小元素。
2)要删除的节点是叶子节点 。处理的方法是释放该节点空间,若是根节点,则令根为NULL。
3 ) 要删除的节点p只有一颗子树。如果p没有节点(即p是根节点),则p的唯一子树的根节点成为新的搜索树的根节点。如果p有父节点pp,则修改pp的指针域,使得它指向p的唯一孩子,然后释放节点p。

  • 以下是我实现的代码
class Solution {
public:
    TreeNode* deleteNode(TreeNode* root, int key) {
        root = deleteNodeitem(root, key);
        return root; 
    }

    TreeNode* deleteNodeitem(TreeNode* root, int key){
        TreeNode* p = root;//the keynode to delete
        TreeNode* pp = root;//parentNode of the keynode
        TreeNode* s;//the node to replace the keynode
        TreeNode* ps;//parent node of replacenode

        //findout the keynode p and its parant node pp
        while((p != NULL)&&(p->val != key)){
            pp = p;
            if(key > p->val){
               p = p->right;    
            }else if(key < p->val){
               p = p->left;
            }
        }
        //can not find the keynode
        if(p == NULL){
            return root;
         }

        //state1 node p have 2 nonull childnode
        if((p->left!=NULL)&&(p->right!=NULL)){
            //find the biggest node in the right tree
            s = p->right;
            ps = p;
            while(s->left != NULL){
                ps = s;
                s = s->left;
            }
           //初始化一个要替换删除节点的节点
           TreeNode q = {s->val};
           q.left = p->left; 
           q.right = p->right;
           //将删除节点指向新构造出来的节点q
           if(pp->left == p){
                pp->left = &q;
            }else if(pp->right == p){
                pp->right = &q;
            }else if(p == root){//要删除的节点就是根节点
                pp->val = q.val;
            }

            //make the s node to delete 找出节点s所对应的父节点
            if(ps != p){
                pp = ps;
            }else if((p == ps)&&(p!=root)){//当s的父节点即为p节点时
                pp = &q;
            }
            p = s;//it must one left child of p if p has child
        }

        //statue2 node p have at most one nonull childnode
       //在只有一个非空子树的情况下,只需要将删除子树的非空子树将其替换就可以了
        TreeNode* pc = p;//the child of p , if it has. 
        if(p->left != NULL){
            pc = p->left;
        }else if(p->right != NULL){
            pc = p->right;
        }else{
            pc = NULL;//p has no child
        }

        if(pp->left == p){
            pp->left = pc;
        }else if(pp->right == p){
            pp->right = pc;
        }else{
            root = pc;//根节点就是需要删除的节点
        }
        return root;
    }
};
  • 利用递归
class Solution {
public:
    TreeNode* deleteNode(TreeNode* root, int key) {
        if (!root) return nullptr;
        if (root->val == key) {
            if (!root->right) {
                TreeNode* left = root->left;
                delete root;
                return left;
            }
            else {
                TreeNode* right = root->right;
                //找出右子树的最小节点  
                while (right->left)
                    right = right->left;
                swap(root->val, right->val);    
            }
        }
       //利用递归找出左右子树中要删除的节点,
       //并利用该节点的右子树中的左子树的最小节点替换掉要被删除的节点
        root->left = deleteNode(root->left, key);
        root->right = deleteNode(root->right, key);
        return root;
    }
};

题目思路整理

  • Find Bottom Left Tree Value 逐层遍历
  • Find Largest Value in Each Tree Row 逐层遍历或前序遍历
  • Find Largest Value in Each Tree Row 利用中序遍历可以得到由小到大的集合
  • Maximum Depth of Binary Tree 后序遍历或逐层遍历
  • Most Frequent Subtree Sum 中序遍历,计算每个节点的值及其左右子树的值之和
  • Invert Binary Tree 后序遍历
  • Binary Tree Tilt 后序遍历求和
  • Sum of Left Leaves 前序遍历,找出左子叶
  • Same Tree 后序或前序遍历节点进行比较
  • Binary Tree Inorder Traversal 中序遍历的实现

未完待续

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,776评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,527评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,361评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,430评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,511评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,544评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,561评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,315评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,763评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,070评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,235评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,911评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,554评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,173评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,424评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,106评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,103评论 2 352

推荐阅读更多精彩内容