版权声明:

本文由Faye_Zuo发布于http://www.cnblogs.com/zuofeiyi/, 本文可以被全部的转载或者部分使用,但请注明出处.

上周学习了数组和链表,有点基础了解以后,这周初步探索了一下深度优先搜索。对于文科生的我来说,这个名词听起来有点可怕。于是我通过leetcode上的一个难度为medium的题目来逐渐认识这个概念的。这道题目是Validate Binary Search Tree(题号为98)。下面我将通过这道题作为引子,整理一下上周学习到的东西。

一、树

这道题目是让我们判断这棵树是否为有效二叉查找树。但是我连树是什么都是一脸茫然。后来通过学习youtube上的视频,对他有逐步了解。树的概念容易理解,现在主要看看它的存储结构(其实我也不知道了解到存储结构有什么用)

树的存储结构如下:

(1)双亲表示法
以一段连续空间来存储树的结点,在每个结点中,附设一个指示器指示其双亲结点到链表中的位置。这样,每个结点除了知道自己是谁以外,还知道自己双亲的位置。
(2)孩子表示法:用多重链表来表示
(3)双亲孩子表示法:上述两种的结合

二、二叉树(Binary Tree)

有了对树的认识以后,二叉树的基本概念就更加容易理解了。

1.概念:每个节点最多有两个节点的结构树,通常子树被称为左子树和右子树。
 
2.二叉树的建立(非常重要)
 
public static void main(String[] args) {
/* Making a tree like :
0
/ \
1 2
/
3
*/ TreeNode root = new TreeNode(0);
root.left = new TreeNode(1);
root.right = new TreeNode(2);
root.left.left = new TreeNode(3);
3.二叉树的遍历
 private static class TreeNode {
    public TreeNode(int x) {    //红色标记部分不是成员函数,是构造函数,主要用来初始化的。所以也不需要有返回值类型。
      this.x = x;
    }
 
    public int x;
    public TreeNode left;//left和right就是变量
    public TreeNode right;
  }
 
  /* pre-order, 先序遍历 */
  private static void traverseTreePreOrder(TreeNode node) {
    if (node != null) {
      System.out.println(node.x);
      traverseTreePreOrder(node.left);
      traverseTreePreOrder(node.right);
    }
  }
 
  /* in-order, 中序遍历 */
  private static void traverseTreeInOrder(TreeNode node) {
    if (node != null) {
      traverseTreeInOrder(node.left);
      System.out.println(node.x);
      traverseTreeInOrder(node.right);
    }
  }
 
  /* post-order, 后序遍历 */
  private static void traverseTreePostOrder(TreeNode node) {
    if (node != null) {
      traverseTreePostOrder(node.left);
      traverseTreePostOrder(node.right);
      System.out.println(node.x);
    }
  } 
 
三、然后我就可以开始看看这道leetcode上的题目了

Validate Binary Search Tree(98)
Given a binary tree, determine if it is a valid binary search tree(BST).
Assume a BST is defined as follows:
(1)The left subtree of a node contains only nodes with keys less than the node’s key.
(2)The right subtree of a node contains only nodes with keys greater than the node’s key.
(3)Both the left and right subtrees must also be binary search trees. 
 
完全不会做啊,于是只有看答案,发现这道题主要要用到递归,思路如下:
 

1.所有的在左边的值必然小于root,所有在右边的值必然大于root。
2.我们要看的是每个节点是不是都满足BST。
3.我们要将这个上限和下限代入递归中。 
 
完整解答的程序:

public class BinarySearchTree {

public static void main(String[] args) {
    TreeNode root=new TreeNode(6);
    root.left=new TreeNode(1);
    root.right=new TreeNode(7);
    root.left.left=new TreeNode(4);
   

    Solution A=new Solution();
    System.out.println(A.isValidBST(root));//为什么new了新对象,还需要将下面的方法改为静态?原因就是看下面的方法是不                        
                                                            //是访问到了对象中特有数据。如果访问到了,加static。
 
  }
  public static class TreeNode{
     int val;
     TreeNode left;
     TreeNode right;
     TreeNode(int x){
      val=x;
    }
  }
 
  public static class Solution{
    public  boolean isValidBST(TreeNode root){
      return isValidBST(root,Double.NEGATIVE_INFINITY,Double.POSITIVE_INFINITY);
    }
   
    public boolean isValidBST(TreeNode p,double min, double max){
      if(p==null)
        return true;
     
      if(p.val<=min||p.val>=max)
        return false;
 
      return isValidBST(p.left,min,p.val)&&isValidBST(p.right,p.val,max);
    }
   
   
  }
  }
 
 
在以上的程序中,我发现我有好多知识点都需要再学习,再补充。整理如下:

1.一些固定需要记住的class
Double.NEGATIVE_INFINITY 表示负数无穷小
Double.POSITIVE_INFINITY 表示正数无穷大
 
2.需要死记的一个算法(写得真的太好了)

public  boolean isValidBST(TreeNode root){
      return isValidBST(root,Double.NEGATIVE_INFINITY,Double.POSITIVE_INFINITY);
    }
   
    public boolean isValidBST(TreeNode p,double min, double max){
      if(p==null)
        return true;
     
      if(p.val<=min||p.val>=max)
        return false;
 
      return isValidBST(p.left,min,p.val)&&isValidBST(p.right,p.val,max);
    }
 
 

3.java中关于静态知识点的讲解
(1)成员变量又叫实例变量,但是静态变量又叫类变量,它是类能直接访问的变量。
(2)成员变量和静态变量的区别:
生命周期:成员变量随着对象的创建而存在,随着对象的回收而释放。静态变量随着类的存在而存在,随着类的消失而消失。
调用方式:成员变量只能被对象调用。静态对象可以被类名调用,也可以被对象调用。
数据存储位置不同:成员变量存储在堆内存的对象中,静态变量存储在方法区中。
(3)static在使用过程中应该注意的细节:
   a.静态方法只能访问静态成员。注意一个常见的报错:无法从静态方法中调用非静态。原因是静态变量会先加载,加载的时候非静态变量中还没有对象。
   b.静态方法中不可以使用this或者super关键字。
   c.主函数一定是静态的。
 (4) 什么时候使用static关键字
静态变量:当对象中所具备的成员变量的值是相同的,这时成员可以被静态修饰。
静态函数:关注一点! 该函数的功能是否访问到对象中的特有数据(特有数据:对象的值)。
 
 4. 递归

(1).递归的最基本思想:自己调用自己
(2).递归必须要有一个出口,即告诉别人什么时候可以不用调用了。否则这个函数会进入一个死循环,直到栈溢出。说白了就是一个if else
(3).递归的特点:耗时间、耗空间(一层一层的调用,再一层一层的返回) 
 
5. 补充面向对象的基本知识

(1)什么是方法:方法就是函数(method)
 
(2)方法的重载:
其实就是同一方法名下,通过参数的不同,可以实现函数的不同功能。
public void tell(int i,int j){}
public void tell(string k){} 
 
本周学习的东西就大致整理分享到这里,下周准备再多练习几道leetcode关于树的题目。希望可以对树这种数据结构有更加深入的理解!
加油!
 

初涉深度优先搜索--Java学习笔记(二)的更多相关文章

  1. 平面上的地图搜索--Java学习笔记(四)

    版权声明: 本文由Faye_Zuo发布于http://www.cnblogs.com/zuofeiyi/, 本文可以被全部的转载或者部分使用,但请注明出处. 这一个月以来,都在学习平面上的地图搜索,主 ...

  2. 深度优先搜索 DFS 学习笔记

    深度优先搜索 学习笔记 引入 深度优先搜索 DFS 是图论中最基础,最重要的算法之一.DFS 是一种盲目搜寻法,也就是在每个点 \(u\) 上,任选一条边 DFS,直到回溯到 \(u\) 时才选择别的 ...

  3. Java学习笔记二十九:一个Java面向对象的小练习

    一个Java面向对象的小练习 一:项目需求与解决思路: 学习了这么长时间的面向对象,我们只是对面向对象有了一个简单的认识,我们现在来做一个小练习,这个例子可以使大家更好的掌握面向对象的特性: 1.人类 ...

  4. 数组排序、递归——(Java学习笔记二)

    升序:      选择排序:         选定一个元素,一次和后面的元素相比较,如果选定的元素大雨后面的比较元素,就交换位置         先出现最小值,最后出现最大值. public stat ...

  5. Java学习笔记二---设置环境变量JAVA_HOME,CLASSPATH,PATH

    1.环境变量包括: JAVA_HOME,CLASSPATH,PATH 2.设置环境变量的目的: 路径搜索,方便查找到jdk的安装路径.方便搜索用到的类文件.方便搜索用到的可执行文件如java,java ...

  6. Java学习笔记二:数据类型II

    GitHub代码练习地址:https://github.com/Neo-ML/JavaPractice/blob/master/Intpractice3.java 一  Java中的数据类型 Java ...

  7. Java学习笔记二十八:Java中的接口

    Java中的接口 一:Java的接口: 接口(英文:Interface),在JAVA编程语言中是一个抽象类型,是抽象方法的集合,接口通常以interface来声明.一个类通过继承接口的方式,从而来继承 ...

  8. Java学习笔记二十七:Java中的抽象类

    Java中的抽象类 一:Java抽象类: 在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就 ...

  9. Java学习笔记二十五:Java面向对象的三大特性之多态

    Java面向对象的三大特性之多态 一:什么是多态: 多态是同一个行为具有多个不同表现形式或形态的能力. 多态就是同一个接口,使用不同的实例而执行不同操作. 多态性是对象多种表现形式的体现. 现实中,比 ...

随机推荐

  1. 文件上传大小js判断

    function fileChange(target) { var fileSize = 0; if (isIE && !target.files) { var filePath = ...

  2. 非旋treap模板

    bzoj3580 非旋转treap 在大神教导下发现split一段区间时先split右边再split左边比较好写 #include <cstdio> #include <cstdli ...

  3. Linux 系统中堆栈的使用方法

    本节内容概要描述了Linux内核从开机引导到系统正常运行过程中对堆栈的使用方式.这部分内容的说明与内核代码关系比较密切,可以先跳过.在开始阅读相应代码时再回来仔细研究. Linux 0.12系统中共使 ...

  4. Linux更改计算机名称

    1.修改:vim /etc/hosts 2.修改:vim /etc/sysconfig/network 3.重启:reboot 如不重启可以输入:hostname  新改的计算机名称,然后su

  5. 其原因可能是堆被损坏,这说明 100BloodCellSegTest.exe 中或它所加载的任何 DLL 中有 Bug。

    这个问题可能是内存空间释放了两次,比如使用cvLoadImage函数时IplImage* img = cvLoadImage(buf.c_str(),1);,注意要释放内存,但不要释放了两次cvRel ...

  6. Problem with "AnyConnect was not able to establish connection to the specified secure gateway."

    Cisco的VPN客户端最近报"AnyConnect was not able to establish connection to the specified secure gateway ...

  7. 浅谈WEB前后端分离

    重审业务逻辑 用过MVC的童鞋都知道业务逻辑(Bussiness Logic),但是大多对这概念又是模棱两可,业务逻辑从来都是这样难以理解,谈论前后端分离之前这个概念非常有必要探讨一下! 在简单的CR ...

  8. ng-table 简单实例

    今天用的AngularJs需要做个分页,于是用ng-table去实现,不过这个官网感觉有点坑,说的不够清楚. 下面实现了一个Demo实力,代码如下: <!DOCTYPE html> < ...

  9. CA认证原理以及实现(上)

    转自:http://yale.iteye.com/blog/1675344 原理基础数字证书为发布公钥提供了一种简便的途径,其数字证书则成为加密算法以及公钥的载体,依靠数字证书,我们可以构建一个简单的 ...

  10. windows下mysql客户端输入中文显示??解决方法

    >>>>>>>>>>>>>>>>>>>> 1.检查并修改mysql的my.ini ...