Divide and Conquer : 分治法

  • 知识点预习:
  1. 分治法: 先让左右子树去解决同样的问题, 然后得到结果之后, 再整合为整棵树的结果。
  2. 遍历法: 通过 前序/ 中序/ 后序 的某种遍历, 游走整棵树, 通过一个全局变量或者传递的参数来记录这个过程中所遇到的点 和 需要计算的结果。
  3. 两种方法的区别: 从实现分治法的角度递归函数, 通常有一个返回值, 遍历法通常没有。
  • pre order
private void traversal( TreeNode root , List<Integer> result ) {
    if( root == null) {
          return ;
    }  
   result.add( root.val);
    traversal( root.left , result);
    traversal( root.right , result);
}
  • in order
private void traversal( TreeNode root , List<Integer> result ) {
      if( root == null) {
            return ;
      }
     traversal( root.left , result) ;
    result.add(root.val);
    traversal( root.right , result);
}
  • post order
private void traversal( TreeNode root , List<Integer> result ) {
      if( root == null) {
            return ;
      } 
      traversal( root.left);
      traversal(root.right);
      result.add(root.val);
}
  1. 二叉树上的分治法
  • 二叉树最大深度
  • 判断平衡二叉树
  • 判断排序二叉树
  1. 用遍历法求 tree 的最大深度 : 遍历法 ; 通常没有 return value
int depth ;
public int maxDepth( TreeNode node) {
      depth = 0;
      travesal( root , 1);
      return depth;
}

private void traversal( TreeNode node , int curDepth) {
      if( node == null) {
            return ;
      }
    depth = depth > curDepth ? depth : curDepth;
    traversal( root.left , curDepth + 1);
    traversal(root.right , curDepth + 1);
}
  1. 用 分治法 通常有 return value
  • 这道题是 max depth 就是 左子树 和 右子树 中较大的depth + 1
public int maxDepth( TreeNode node) {
   return  helper( root);
}

private int helper(TreeNode root ) {
      if(root == null) {
            return 0;
      }
      int left = helper( root.left) + 1 ;
      int right = helper(root.right )+ 1;
      return Math.max(left , right);
}
  • 判断平衡二叉树 : 分治法
  1. 平衡就 return 高度, 不平衡就return 不平衡
  • 一边返回高度一边 判断是否是 平衡二叉树
private int NOT_BALANCED = -1;
public boolean isBalanced( TreeNode root) {
    return  maxDepth(root) != NOT_BALANCED;
}
private int maxDepth( TreeNode root ) {
    if( root == null) {
        return 0;
    }
    int left = maxDepth( root.left);
    int right = maxDepth( root.right);
    if( left == NOT_BALANCED || right == NOT_BALANCED) {
          return NOT_BALANCED;
    }
    if( Math.abs( left - right) > 1) {
          return NOT_BALANCED;
    }
    return Math.max( left , right) + 1;
}
  • 判断二叉搜索树: 遍历法 vs 分治法
    1 . 遍历法: 因为 BST 树的特性, inorder 的遍历法 就是一个 上升序列, 所以可以用 遍历法解决 。 遍历法 没有 return value, 但是有golbal 参数
private TreeNode lastNode;
private boolean isValid;
public boolean isValidBST( TreeNode root) {
      isValid = true;
      lasNode = null;
      inorderTraverse(root);
      return isValid;
}
private void inorderTraverse( TreeNode root) {
      if( root == null ) {
            return ;
      }
      inorderTraverse( root.left);
      if( lastNode != null &&  lastNode.val >= root.val ) {
              isValid = false;
              return ;
       }
      lastNode = root;
      inorderTraverse( root.right);
} 

  1. 分治法 :
    ❌做法: 遍历到每个节点, 然后比较左右孩子
    正确做法: current root 要比 左子树最大的值 ,还要大; 比右子树 的最小值还要小, 所以这道题 要return 2 个结果, min 和 max
class ResultType{
    boolean isValid;
    TreeNode minNode , maxNode;
    public ResultType(boolean default) {
          this.isValid = default;
          this.minNode = null;
          this.maxNode = null
    }
}

public boolean isValid( TreeNode root) {
        return divideConquer(  root).isValid;
}

private ResultType divideConquer( TreeNode root ) {
    if(root == null ) {
          return new ResultType(true);
    }
    ResultType left = divideConquer( root.left);
    ResultType right = divideConquer( root.right);
    // 判断错误
    if( ! left.isValid || !right.isValid) {
          return new ResultType(false);
    }
   if( left.maxNode != null && root.val  <= left.maxNode) {
         return new ResultType(false);
    }
    if( right.minNode != null  && root.val >=right.minNode ) {
          return new ResultType(false);
    }
    // bst
    ResultType res = new ResultType(true);
    res.minNode = left.minNode == null ? root : left.minNode;
    res.maxNode = right.maxNode == null ? root : right.maxNode;
    return res; 
}
  • 知识点: 递归, 回溯 和 搜索

  • 递归 : 1 . 是一种由大化小, 由小化无的解决问题的算法。类似的算法还有动态规划。
    2. 自己调用自己

  • 非递归:迭代法

    1. 什么是搜索(search)
    • 搜索分为 深度优先搜索( DFS)和 宽度优先搜索(BFS); 搜索通常是一种 类似于枚举的算法。
  • 回溯法 就是 深度优先搜索。 回溯是 深度优先搜索过程中的一个步骤。
    例如: 我们在进行全子集问题的搜索时 ,假如 当前的集合 是{1, 2} 代表我正在 寻找以{
    1, 2} 开头的所有集合。 那么下一步 回去寻找 { 1, 2, 3} 开头的所有集合 , 当然当我们找完所有以{1,2,3} 开头的集合时, 我们需要把3 从集合中 删除, 回到{1,2} . 然后 再把 4 放进去, 寻找以{1,2,4} 开头的集合。 这个把 3 删掉回到 {1, 2} 的过程, 就是回溯。

  • bst定义: (BST)

  1. 若左子树 不为空, 则左子树上所有的节点值均小于 或等于它的根结点
  2. 若右子树不为空, 则右子树上所有的节点值均大于根节点值
  3. 左右子树也为二叉搜索树
  • 平衡搜索树

    1. 空树可以是 balanced binary tree ; 2. 左右子树高度差不超过 <=1
  • 特性 :高度为 o(long n)

  • 最大作用: 保证查找的最坏时间复杂度为 o(log n)

** 额外拓展:

  1. morris 算法实现 o(1) 额外空间对二叉树进行 先序遍历
  2. 用非递归(non-recusrion/ iteratioin)的方式实现二叉树的 前序遍历, 中序遍历 和 后序遍历
  3. BST 的增删改查
  4. 平衡排序二叉树(balanced binary search tree)以及 TreeSet / TreeMap 的使用
  • 2 非递归的方式实现二叉树遍历
  1. 先序遍历 preorder
public List<Integer> preorderTraversal(TreeNode root) {
    Stack<Integer> stack = new Stack<>();
     List<Integer> result = new ArrayList<>();
    TreeNode p = root;
    stack.push( root);
    while( !stack.empty()  || p != null ) {
          if( p != null ) {
                stack.push( p);
                result.add( p.val );
                p = p.left;
          } else {
              TreeNode node = stack.pop();
              p = p.right;
          }
    }
    return result;
}
  1. 中序遍历
public List <Integer> inorderTraversal(TreeNode root) { 
      Stack<TreeNode> stack = new Stack<>();
      List<Integer>  result  = new ArrayList<>();
      TreeNode p = root;
      while( p != null ||  !stack.isEmpty() ) {
              while( p != null) {
                    stack.push(p );
                    p = p.left;
              }
             TreeNode node  = stack.pop();
             result.add( node.val);
             p = node.right
      }
      return result;
}
  1. 后序遍历
public List<Integer> postorderTraversal(TreeNode root) { 
    Stack< TreeNode> stack = new Stack<>();
    List<Integer> result = new LinkedList<>();
    TreeNode p = root ;
    while( p != null || !stack.isEmpty() ) {
        if( p != null) {
                stack.add( p );
                result.add( 0 , p);
                p = p.left;
         } else {
              TreeNode node = stack.pop();
               p = node.right;
          }
    }
return result;
}
  1. morris 算法实现 o(1) 额外空间对二叉树进行 先序遍历
    preorder
public List<Integer> preorderTraversal(TreeNode root) {
    List<Integer> nums = new ArrayList<>();
    TreeNode cur = null;
    while( root != null) {
          if( root.left != null) {
              cur = root.left;
          }
          while( cur.right != null && cur.right != root) {
                  cur = cur.right ;  
          }
          if( cur.right == root) { // 撤销钩子
                  cur.right = null;
                  root = root.right;
          } else { // cur.right == null
                 nums.add( root.val);
                 cur.right = root;
                cur = root.left ;
          }
    } else {// root.left == null
            nums.add(root.val);
            root = root.right;
    }
    return nums;
}
  1. inorder
public List<Integer> inorderTraversal(TreeNode root) { 
       List<Integer> nums = new ArrayList<>();
       TreeNode cur = null;
        while( root != null) {
                if( root.left != null ) {
                    cur = root.left ;
                    while( cur.right != null && cur.right != root) {
                            cur = cur.right ;
                    } 
                    if( cur.right == root) {
                            nums.add(root);  
                            root = root.right;
                            cur.right = null;
                      } else { // cur.right == null
                              cur.right = root;
                              root = root.left;
                      }
                    
                } else {
                        nums.add(root.val);
                        root = root.right;
                 }
        }    
}
  1. postorder
public List<Integer> inorderTraversal(TreeNode root) { 
      TreeNode cur = null;
      List<Integer> nums = new ArrayList<>();
      while( root != null) {
          if( root.right != null ) {
                cur =root.right;
                while( cur.left != null && cur.left != root) {
                        cur = cur.left;
                }
                if( cur.left == root) {
                      cur.left = null;
                      root = root.left;
                } esle {
                      cur.left = root;
                      nums.add(root.val);
                      root = root.right;
                }
          } else {
              nums.add(root.val);
              root = root.left; 
          }
      }
Collections.reverse( nums);
 return nums;
}

BST 的增删改查

  • 在求 BST 的 时间复杂度追求 : O(log N)
  1. closestValue : 找到 pre , suc 然后比较

  2. two sum: 用 pre , suc + 双指针

  3. BST 里面要知道如何求 predecessor 和 successor
    predecessor:

private TreeNode inorderPredecesor (TreeNode root , TreeNode node  ) {
    TreeNode res  = null;
    while( root != null) {
         if( root.val  > node.val) {
                root = root.left;
          } else { // root.val <= node.val 
               res = root;
                root = root.right;
          }
    }
    return res; 
}

successor

private TreeNode inorderSuccessor ( TreeNode root , TreeNode node ) {
    TreeNode res = null;
     while( root != null) {
        if( root.val < node.val ) {
               root = root.right ;
        } else { // root.val >= node.val
              res = root;
              root = root.left ;
      }
    }
    return res;
}

BST 删除

BST 插入

BST 修改

BST 查找

======================================================================

1.Kth Largest Element in an Array(215. leetcode)
public int findKthLargest( int [] nums, int k){
    if( nums == null || nums.length == 0)
        return -1;
    return quickSelect( nums, 0, nums.length - 1, nums.length - k + 1);
}

private int quickSelect(int [] nums , int start , int end , int k ) {
    if(start == end) return nums[ start];
    int mid = start + (end - start )/2 ; 
    int pivot = nums[ mid];
    int i =start , j = end;
    while( i <= j ){
        while( i <= j && nums[ i ] < pivot) i ++;
        while( i <= j && nums[ j] < pivot ) j -- ;
        if( i <= j) {
            int tmp = nums[i];
            nums[ i] = nums[ j];
            nums[ j ] = tmp;
            i ++;
            j -- ;
        } / * if */
    } / * while */
   if ( start + k - 1 <= j ){
        return quickSelect( nums, start, j , k );
   } else if ( start + k - >=i ) {
        return quickSelect( nums, i , end , k - i + start);
   }
   return nums[ j + 1]; 
}
2. Closest Binary Search Tree Value II (google )
  • getStack() : 到达离 target 最近的 node 时, 所经过的点
  • moveUpper( stack) : 根据 stack , 挪动到 next node
  • moveLower( stack) : 根据 stack , 挪动到 prev node
    有了这些函数之后, 就可以把整棵树当做一个数组一样来处理, 只不过每次 i++ 的时候要用
    moveUpper , i -- 的时候要用 moveLower
public List<Integer> closestKValues(TreeNode root, double target, int k) {
      List< Integer> values = new ArrayList<>();
      if( k == 0 || root == null) {
            return values;
      }      
      Stack<TreeNode> lowerStack = getStack();
      Stack<TreeNode> upperStack = new Stack<>();
      upperStack.addAll( lowerStack) ;
      if( target < lowerStack.peek().val ) {
            moveLower( lowerStack);
      }  else {
          moveUpper( upperStack);
      }
      for( int i =0 ; i < k ; i++) {
            if(  lowerStack.isEmpty()  || target - lowerStack.peek().val  > upperStack().peek().val - target) {    values.add(upperStack.peek().val )
                  moveUpper( upperStack);
            } else {
                  values.add( lowerStack.peek().val );
                  moveLower( lowerStack);
            }
      }
    return values;
}

private Stack<TreeNode >  getStack( TreeNode root , double target ) {
      Stack<TreeNode > res = new Stack <>();
      while( root != null) {
             res.push( root );
             if( root.val >  target) {
                  root = root.left ;
              } else { // root.val <= target
                     root = root.right;
              }
      }
  return res;
}

private void moveUpper( Stack<TreeNode> stack) {
       TreeNode node = stack.peek();
        if( node.right == null) {
              node = stack.pop();
              while( !stack.isEmpty() || stack.peek().right == node ) {
                      node = stack.pop();
              }
              return ;
        }
        node = node.right;
        while( node != null) {
              stack.add(node);
              node = node.left;
        }
}

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

推荐阅读更多精彩内容