版权声明:

本文由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. C#动态规划查找两个字符串最大子串

     //动态规划查找两个字符串最大子串         public static string lcs(string word1, string word2)         {            ...

  2. *** wechat-php-sdk 微信公众平台php开发包

    wechat-php-sdk 微信公众平台php开发包,细化各项接口操作,支持链式调用,欢迎Fork此项目weixin developer SDK. 项目地址:https://github.com/d ...

  3. PHP socket上传文件图片

    最近了解了下下socket方面的东西,想做一个socket上传文件的例子. 在网上搜了搜代码执行后,图片数据传输了一半,图片的下半部分是灰色的.然后就自己仿着搜来的代码和php.net 中socket ...

  4. Eclipse代码注释模板

    <?xml version="1.0" encoding="UTF-8"?><templates><template autoin ...

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

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

  6. FFT与乒乓球

    刚刚打乒乓球的时候,看到一个旋球.想起<傅里叶分析之掐死教程>: “正弦曲线波叠加出一个带90度角的矩形波来” 我们把多个旋叠加在一起,是不是就可以让这个球跳舞了呢?

  7. Java 之 常用类(一)

    1.字符串: a.分类:String.StringBuffer.StringBuilder b.特殊:①String是唯一一个可以直接用常量赋值的引用数据类型 ②String的常量也是一个对象 (即 ...

  8. python strip()函数 介绍

    python strip()函数 介绍,需要的朋友可以参考一下   函数原型 声明:s为字符串,rm为要删除的字符序列 s.strip(rm)        删除s字符串中开头.结尾处,位于 rm删除 ...

  9. ubuntu14.04 安装 搜狗输入法

    1.安装或者更新fcitx libssh2-1:sudo  apt-get install fcitx libssh2-1; 2.搜索是否安装成功: dpkg -l | grep fcitx ; dp ...

  10. hdu2662

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2662 莫名其妙写了一个题,感觉还是很有价值的记录一下. 题目大意:给两个互质的数,求用无限个它们不能组 ...