一、中序线索化

二叉树节点定义:

class TreeNode {
int val = 0;
TreeNode left = null;
TreeNode right = null;
int isleftChild = 1;//0:线索 1:左孩子
int isrightChild = 1; public TreeNode(int val) {
this.val = val;
}
}

在中序遍历的过程中完成线索化

   //preNode始终指向中序序列中前一个访问的节点
private static TreeNode preNode = null;//只能用全局变量的形式来记录。 //curNode始终指向中序序列中当前访问的节点
public static TreeNode inIOrderThread(TreeNode curNode) {
if (curNode == null)
return null; //线索化左子树
InClueTree(curNode.left); //线索化当前节点
if (curNode.left == null) {
curNode.isleftChild = 0;
curNode.left = preNode;
}
if (preNode != null && preNode.right == null) {
preNode.isrightChild = 0;
preNode.right = curNode;
}
preNode = curNode;// //线索化右子树
InClueTree(curNode.right); return curNode;
}

遍历中序线索二叉树

public static void inOrderThreadTravel(TreeNode root) {
while (root != null) {
System.out.print(root.val + " ");
//存在线索,root的直接后继就是root.rigth;
if (root.isrightChild == 0) {
root = root.right;
}
//不存在线索的时候一定存在孩子节点,则root的直接后继就是其右子树的中序第一个元素。
else {
root = root.right;
while (root.left != null && root.isleftChild == 1) {
root = root.left;
}
}
}
}

二、表达式求值

有两个表达式

  (4 + (13 / 5)) = 6        (a)

  ((2 + 1) * 3) = 9    (b)

对应的两个表达式树(a)(b)

特点:数字都在叶子节点,运算符都在根节点。

    +                                   *
/ \ / \
4 / + 3
/ \ / \
13 5 2 1   (a)                    (b)

来看一下表达式树的前中后三种顺序遍历结果;

中序:

  4 + 13 / 5  ---  (a)

  2 + 1 * 3    ---  (b)

可以看出,表达式树的中序遍历结果就是正常书写顺序,也是计算机可以直接求解的方式。

后序:

  4 13 5 / +  ---  (a)

  2 1 + 3 *    ---  (b)

此时遍历结果非书写顺序计算机也不能直接求解,非要按照这个顺序用计算机求解,怎么办?

解决方案:栈

  按照遍历顺序对元素如下的操作:

    1、如果元素是数字,入栈

    2、如果元素是操作符不入栈,反而弹栈两个元素a,b;将a作为运算符的左操作数,b作为右操作数计算得到结果c;将结果c入栈。

    3、重复上述操作,直到没有元素时,此时栈中一定只有一个元素,将其返回。

前序:

  + 4 / 13 5  ---  (a)

  * + 2 / 3    ---   (b)

此时遍历结果非书写顺序计算机也不能直接求解,非要按照这个顺序用计算机求解,怎么办?

解决方案:栈

  与后序列操作类似,只不过按照遍历顺序的逆序,为什么是这样呢?

   因为:栈的特点可以暂存之前遇到的信息,在后续操作中可以从栈中取出之前保存的信息;

      四则运算符都是二元运算符,因此一次计算的顺利完成需要3个信息,两个数字,一个运算符号;

        因此遇到数字时候压栈,遇到操作符时候不压栈,然而弹出两个元素进行计算,这是合理的

      而观察表达式树我们发现,叶子节点全都是数字,跟节点全都是操作符号,在进一步可以这么想,父节点都是操作符,孩子节点都是数字(当然直观来看不是这样的,如表达式树(a)中根节点“+”的右孩子明明是“/”;其实在根节点“+”真正计算的时候,13 和 5的父节点“/”早就是新的数字了);结合树的遍历特点,要么遍历完孩子节点才遍历根节点(后序),要么遍历完孩子节点才遍历根节点(前序),总之,孩子节点(数字)总在父节点(符号)的一侧。不管是先序还是后序,我们都统一为先处理孩子节点,再处理父节点,后序顺序中,孩子节点刚好在父节点之前,因此不做顺序调整,而先序遍历的时候,孩子节点均在父节点之后,因此需要逆序调整。

import java.util.Stack;

public class Solution {
public int evalRPN(String[] tokens) {
int res = 0;
if (tokens == null || tokens.length == 0)
return res;
int len = tokens.length;
Stack<Integer> stack = new Stack();
int i = 0;
for (; i < len; i++) {
if (isNum(tokens[i]))
stack.push(Integer.parseInt(tokens[i]));
else {
int a = stack.pop();
int b = stack.pop();
//除法有顺序要求哦
stack.push(operator(b, a, tokens[i]));
}
}
if (i == len)
return stack.pop();
return res;
} private boolean isNum(String str) {
boolean b = false;
try {
Integer.parseInt(str);
b = true;
} catch (Exception e) {
}
return b;
} private int operator(int a, int b, String s) {
int res = 0;
switch (s) {
case "+": {
res = a + b;
break;
}
case "-": {
res = a - b;
break;
}
case "*": {
res = a * b;
break;
}
case "/": {
res = a / b;
break;
}
}
return res;
}
}

     

最新文章

  1. weblogic配置数据源
  2. Blob初探
  3. Images.xcassets
  4. js判断当前的访问是手机还是电脑
  5. 【mysql】使用tpcc-mysql进行压力测试
  6. android.mk文件里的通配符
  7. 最近的学习的linux命令笔记
  8. memcached原理全面剖析
  9. POJ 3419 Difference Is Beautiful(RMQ+二分 或者 模拟)
  10. 单Js 的重力游戏开发
  11. 机器学习之支持向量机(四):支持向量机的Python语言实现
  12. Elasticsearch安装详解
  13. 修改 bug 总结
  14. faster-rcnn原理讲解
  15. Python001-操作MSSQL(Microsoft sql server)基础示例(一)
  16. Java学习笔记(6)
  17. spring boot整合websocket
  18. shell脚本--shift参数左移
  19. Hdu-1358Period(KMP算法之next数组的应用)
  20. struts下载文件

热门文章

  1. SGU 132. Another Chocolate Maniac 状压dp 难度:1
  2. 几句话概括理查德成熟度模型(RESTful)
  3. WebService 初步入门的理解
  4. [Qt] QLineEdit 仿QQ签名框
  5. 【Beanstalkd】Beanstalkd消息队列的安装与使用
  6. 【java多线程】ConCurrent并发包 - Lock详解
  7. 关于 unable to load shared object &#39;C:\Program Files\R\R-3.0.3\library\stats\libs\i386\stats.dll&#39;
  8. 在WPF中使用CefSharp嵌入浏览器(转)
  9. hasura graphql server (haskell)构建
  10. 使用docker 部署graylog集群