目录

1 问题描述

2 解决方案

 


1 问题描述

在了解最优二叉查找树之前,我们必须先了解何为二叉查找树?

引用自百度百科一段讲解:

二叉排序树(Binary Sort Tree)又称二叉查找树(Binary Search Tree),亦称二叉搜索树

二叉排序树或者是一棵空树,或者是具有下列性质的二叉树

(1)若左子树不空,则左子树上所有结点的值均小于或等于它的根结点的值;

(2)若右子树不空,则右子树上所有结点的值均大于或等于它的根结点的值;

(3)左、右子树也分别为二叉排序树;

在二叉查找树的基础上,引出了一个最优二叉查找树的问题:它在查找树中所有节点的平均键值比较次数是最低的。(PS:如若对于最优二叉查找树的定义理解还是有点模糊,可以参考本文最后给出的参考资料中的链接)


2 解决方案

本文具体编码思想参考自《算法设计与分析基础》第三版,具体如下(PS:对于文中的具体思想,楼主自己是前后看了三四遍才整明白其具体思想,竟无语吟噎......,如若对于下面贴出的书中介绍无法理解,可以参考文末给出的参考资料的链接中,一位网友的博客讲解哦):

具体代码如下:

package com.liuzhen.chapter8;

public class OptimalBST {
/*
* 参数P:表示1~n个节点的查找概率。其中P[0] = 0,无意义
* 函数功能:返回在最优BST中查找的平均比较次数主表C[][],以及最优BST中子树的根表R
*/
public void getBestTree(double[] P) {
int lenP = P.length;
double[][] C = new double[lenP+1][lenP]; //保存最有BST的成功查找的平均比较次数
int[][] R = new int[lenP+1][lenP]; //保存最优BST中子树的根表R
for(int i = 1;i < lenP;i++) {
C[i][i] = P[i];
R[i][i] = i;
} for(int d = 1;d < lenP-1;d++) {
for(int i = 1;i < lenP-d;i++) {
int j = i + d;
double minval = Double.MAX_VALUE; //以double类型的最大值,表示minval趋向无穷大
int kmin = 0;
for(int k = i;k <= j;k++) {
if(C[i][k-1] + C[k+1][j] < minval) {
minval = C[i][k-1] + C[k+1][j];
kmin = k;
}
}
R[i][j] = kmin;
double sum = P[i];
for(int s = i+1;s <= j;s++)
sum += P[s];
C[i][j] = minval + sum;
}
} System.out.println("在最优BST中查找的平均比较次数依次为:");
for(int i = 1;i < C.length;i++) {
for(int j = 0;j < C[0].length;j++)
System.out.printf("%.1f\t",C[i][j]);
System.out.println();
} System.out.println("在最优BST中子树的根表R为:");
for(int i = 1;i < R.length;i++) {
for(int j = 0;j < R[0].length;j++)
System.out.print(R[i][j]+"\t");
System.out.println();
}
} public static void main(String[] args) {
OptimalBST test = new OptimalBST();
double[] P = {0,0.1,0.2,0.4,0.3};
test.getBestTree(P);
} }

运行结果:

在最优BST中查找的平均比较次数依次为:
0.0 0.1 0.4 1.1 1.7
0.0 0.0 0.2 0.8 1.4
0.0 0.0 0.0 0.4 1.0
0.0 0.0 0.0 0.0 0.3
0.0 0.0 0.0 0.0 0.0
在最优BST中子树的根表R为:
0 1 2 3 3
0 0 2 3 3
0 0 0 3 3
0 0 0 0 4
0 0 0 0 0

参考资料:

1.《算法设计与分析基础》第3版  Anany Levitin 著 潘彦 译

2.动态规划方法生成最优二叉查找树

算法笔记_053:最优二叉查找树(Java)的更多相关文章

  1. 算法笔记_023:拓扑排序(Java)

    目录 1 问题描述 2 解决方案 2.1 基于减治法实现 2.2 基于深度优先查找实现 1 问题描述 给定一个有向图,求取此图的拓扑排序序列. 那么,何为拓扑排序? 定义:将有向图中的顶点以线性方式进 ...

  2. 算法笔记_228:信用卡号校验(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 当你输入信用卡号码的时候,有没有担心输错了而造成损失呢?其实可以不必这么担心,因为并不是一个随便的信用卡号码都是合法的,它必须通过Luhn算法来验证 ...

  3. 算法笔记_138:稳定婚姻问题(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 何为稳定婚姻问题? 有一个男士的集合Y = {m1,m2,m3...,mn}和一个女士的计划X = {n1,n2,n3,...,nn}.每一个男士有 ...

  4. 算法笔记_137:二分图的最大匹配(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 何为二分图的最大匹配问题? 引用自百度百科: 首先得说明一下何为匹配: 给定一个二分图G,在G的一个子图M中,M的边集{E}中的任意两条边都不依附于 ...

  5. 算法笔记_132:最大流量问题(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 何为最大流量问题? 给定一个有向图,并为每一个顶点设定编号为0~n,现在求取从顶点0(PS:也可以称为源点)到顶点n(PS:也可以称为汇点)后,顶点 ...

  6. 算法笔记_040:二进制幂(Java)

    目录 1 问题描述 2 解决方案 2.1 从左至右二进制幂 2.2 从右至左二进制幂   1 问题描述 使用n的二进制表示,计算a的n次方. 2 解决方案 2.1 从左至右二进制幂 此方法计算a的n次 ...

  7. 算法笔记_014:合并排序(Java)

    1 问题描述 给定一组数据,使用合并排序得到这组数据的非降序排列. 2 解决方案 2.1 合并排序原理简介 引用自百度百科: 合并排序是建立在归并操作上的一种有效的排序算法.该算法是采用分治法(Div ...

  8. 算法笔记_233:二阶魔方旋转(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 魔方可以对它的6个面自由旋转. 我们来操作一个2阶魔方(如图1所示): 为了描述方便,我们为它建立了坐标系. 各个面的初始状态如下:x轴正向:绿x轴 ...

  9. 算法笔记_227:填写乘法算式(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 观察下面的算式: * * × * * = * * * 它表示:两个两位数字相乘,结果是3位数.其中的星号(*)代表任意的数字,可以相同,也可以不同, ...

随机推荐

  1. RHEL/CentOS网络相关的配置文件路径

    /etc/sysconfig/network-scripts/ifcfg-ens33 //IP地址,子网掩码等配置文件 /etc/sysconfig/network-scripts/ifcfg-lo ...

  2. Linux基础系列-Day6

    Samba服务(基于CentOS 7.0) Samba是在Linux和UNIX系统上实现SMB协议的一个免费软件,由服务器及客户端程序构成,Samba主要用于Linux或UNIX和Windows系统之 ...

  3. Sd - 数据库开发调优

    尤其是Sql写法上的技巧,以及常见Sql的写法

  4. BZOJ 4802 欧拉函数(Pollard_Rho)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=4802 [题目大意] 已知N,求phi(N),N<=10^18 [题解] 我们用P ...

  5. 【Trie】MIPT-2016 Pre-Finals Workshop, Taiwan NTU Contest, Sunday, March 27, 2016 Problem B. Be Friends

    题意:一个n个点的完全图,点带权,边权是两端点点权的异或值.问你最小生成树. 一个性质,把所有点按照二进制最高位是否为1划分为2个集合,那么这两个集合间只会有一条边.可以递归处理. 把所有点建成01T ...

  6. 【数论】【Polya定理】【枚举约数】【欧拉函数】【Java】poj2154 Color

    你随便写一下出来,发现polya原理的式子里面好多gcd是相同的,gcd(n,i)=k可以改写成gcd(n/k,i/k)=1,也就是说指数为k的项的个数为phi(n/k),就很好求了,最后除的那个n直 ...

  7. 【DFS】POJ3009-Curling 2.0

    [题目大意] 给出一张地图,一旦往一个方向前进就必须一直向前,直到一下情况发生:(1)碰到了block,则停在block前,该block消失:(2)冲出了场地外:(3)到达了终点.改变方向十次以上或者 ...

  8. JavaScript之引用类型(Object类型)

    ECMAScript提供了很多原生的引用类型,以便开发人员进行常见的计算任务. 对象是某一个特定引用类型的的实例. Object类型 用的最多.虽然这个Object实例不具备多少功能,但是在应用程序的 ...

  9. [转]JSP中常见的Tomcat报错错误解析(一)

    1**:请求收到,继续处理2**:操作成功收到,分析.接受3**:完成此请求必须进一步处理4**:请求包含一个错误语法或不能完成5**:服务器执行一个完全有效请求失败 100——客户必须继续发出请求1 ...

  10. SpringBoot使用Gradle构建war包

    Spring Boot默认将应用打包成可执行的jar包.有时候需要打包成war包部署在tomcat等容器.下面简单介绍下打包的步骤. 一.修改gradle.build文件 1.1 添加如下配置 app ...