>歃血~红颜

导航

< 2025年1月 >
29 30 31 1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31 1
2 3 4 5 6 7 8

统计

(源码,具体的细节请查阅相关资料)哈弗曼树的构造以及非递归遍历树

 


写了一点haffman树的创建和二叉树的非递归遍历.

  • 如果编写代码的时候出现了,思维断点,可以借鉴一下, 避免浪费一些不必要的时间.
我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占 位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我 是占位符我是占位符我是占位符我是占位符我是占位符我是 占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符
卧室占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符 我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符 我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符我是占位符

 


  

 

复制代码
  1 package cc.algorithms;
  2 
  3 import java.util.ArrayList;
  4 import java.util.LinkedList;
  5 import java.util.List;
  6 
  7 public class HaffManTree<E> {
  8     private Node<E> root ;
  9     public static class Node<E> {
 10         E data ;
 11         double weight ;
 12         Node<E> left, right;
 13         Node<E> parent ;
 14 
 15         public Node() {
 16             this.left = this.right = this.parent = null ;
 17         }
 18         public Node(E data, double weight) {
 19             this.data = data ;
 20             this.weight = weight ;
 21         }
 22 
 23         @Override
 24         public String toString() {
 25             if (data == null) {
 26                 return "[data=null, weight=" + this.weight + "]" ;
 27             }
 28             else
 29                 return "[data=" + this.data + ", weight=" + this.weight + "]";
 30         }
 31     }
 32 
 33     public static void main(String[] args) {
 34         List<Node<String>> nodes = new ArrayList<>() ;
 35         nodes.add(new Node("A", 99)) ;
 36         nodes.add(new Node("B", 21)) ;
 37         nodes.add(new Node("C", 211)) ;
 38         nodes.add(new Node("D", 26)) ;
 39         nodes.add(new Node("E", 24)) ;
 40         nodes.add(new Node("F", 23)) ;
 41 
 42         HaffManTree<String> hf = new HaffManTree() ;
 43         hf.create(nodes) ;
 44         System.out.println("前序:") ;
 45         hf.prefByStack(); ;
 46         System.out.println("中序");
 47         hf.inByStack() ;
 48         System.out.println("后序:");
 49         hf.hostByStack() ;
 50         System.out.println("层序");
 51         hf.floorByQueen() ;
 52     }
 53 
 54 
 55     /**
 56      * 创建haffman树
 57      */
 58     public void create(List<Node<E>> nodes) {
 59         //只要列表中有2个以上的节点
 60         while(nodes.size() > 1) {
 61             shellSort(nodes);
 62 
 63             Node<E> min1 = nodes.get(nodes.size() - 1) ;
 64             Node<E> min2 = nodes.get(nodes.size() - 2) ;
 65 
 66             //生成父节点
 67             Node<E> parent = new Node(null, min1.weight + min2.weight) ;
 68 
 69             parent.left = min1 ;
 70             parent.right = min2 ;
 71             min1.parent = parent;
 72             min2.parent = parent ;
 73             //从列表中删除两个最小节点
 74             nodes.remove(nodes.size() - 1) ;
 75             nodes.remove(nodes.size() - 1);
 76 
 77             nodes.add(parent) ;
 78         }
 79         this.root =  nodes.get(0) ;
 80     }
 81 
 82     /**
 83      * 希尔排序--从大到小排序
 84      */
 85     public void shellSort(List<Node<E>> nodes) {
 86         int N = nodes.size() ;
 87         int h = 1 ;
 88         while(h < N / 3) h = 3 * h + 1 ;
 89         while(h >= 1) {
 90             for(int i = h ; i < N ; ++i) {
 91                 for(int j = i ;
 92                     j >= h && nodes.get(j).weight > nodes.get(j - h).weight ;
 93                     j -= h) {
 94                     Node temp = nodes.get(j) ;
 95                     nodes.set(j, nodes.get(j - h)) ;
 96                     nodes.set(j - h, temp) ;
 97                 }
 98             }
 99             h /= 3 ;
100         }
101         for(Node<E> x : nodes) {
102             System.out.println(x);
103         }
104         System.out.println("*********************************") ;
105     }
106     /**
107      * 前序遍历 --------- 递归
108      */
109     public void pre() {
110         System.out.println(this.root) ;
111         prefOrder(this.root);
112     }
113     private void prefOrder(Node<E> node) {
114         if(node != null) {
115             System.out.println(node) ;
116             prefOrder(node.left);
117             prefOrder(node.right);
118         }
119     }
120     public void prefByStack() {
121         ArrayList<Node<E>> st = new ArrayList<>() ;
122         Node<E> p = this.root ;
123 
124         while(p != null || !st.isEmpty()) {
125             while(p != null) {
126                 System.out.println(p) ;
127                 st.add(p) ;
128                 p = p.left ;
129             }
130             if(!st.isEmpty()) {
131                 p = st.remove(st.size() - 1) ;
132                 p = p.right ;
133             }
134         }
135     }
136     public void inByStack() {
137         ArrayList<Node<E>> st = new ArrayList<>() ;
138         Node<E> p = this.root ;
139 
140         while(p != null || !st.isEmpty()) {        //站不空,或者当前节点不空
141             while(p != null) {
142                 st.add(p) ;
143                 p = p.left ;
144             }
145             if(!st.isEmpty()) {                //pop,并访问,然后换右子树
146                 p = st.remove(st.size() - 1) ;
147                 System.out.println(p) ;
148                 p = p.right ;
149             }
150         }
151     }
152     //需要判断右子树是否访问过
153     public void hostByStack() {
154         ArrayList<Node<E>> st = new ArrayList<>() ;
155         Node<E> p = this.root ;
156         Node<E> pre = null ;
157 
158         while(p != null || !st.isEmpty()) {
159             while(p != null) {
160                 st.add(p) ;
161                 p = p.left ;
162             }
163             if(!st.isEmpty()) {
164                 //查看栈顶元素,判断右子树是否访问过<br>
165                 //访问过,就访问当前节点<br>
166                 //否则,访问右子树
167                 p = st.get(st.size() - 1) ;
168                 if(p.right == null || p.right == pre) {
169                     p = st.remove(st.size() - 1) ;
170                     System.out.println(p) ;
171                     //更新
172                     pre = p ;
173                     p = null ;
174                 } else {
175                     p = p.right ;
176                 }
177             }
178         }
179     }
180 
181     /**
182      * 总结:
183      *  1.前序,第一次碰到节点的时候访问<br>
184      *  2.中序,第二次碰到节点的时候访问,<br>
185      *  3.后序,第三次碰到节点的时候访问<br>
186      */
187 
188 
189     //层次遍历---即广度优先遍历
190     public void floorByQueen() {
191         LinkedList<Node<E>> queue = new LinkedList<>() ;
192         Node<E> p = this.root ;
193         //将根入队
194         if(p != null) {
195             queue.addLast(p); ;
196             while(!queue.isEmpty()) {
197                 p = queue.removeFirst() ;
198                 System.out.println(p) ;
199                 //将左右子树加入队列中
200                 if(p.left != null) {
201                     queue.addLast(p.left);
202                 }
203                 if(p.right != null) {
204                     queue.addLast(p.right) ;
205                 }
206             }
207         }
208     }
209 }
复制代码

 

 

 


posted on   一杯酒的故事  阅读(210)  评论(0编辑  收藏  举报

编辑推荐:
· Linux系统下SQL Server数据库镜像配置全流程详解
· 现代计算机视觉入门之:什么是视频
· 你所不知道的 C/C++ 宏知识
· 聊一聊 操作系统蓝屏 c0000102 的故障分析
· SQL Server 内存占用高分析
阅读排行:
· 盘点!HelloGitHub 年度热门开源项目
· DeepSeek V3 两周使用总结
· 02现代计算机视觉入门之:什么是视频
· C#使用yield关键字提升迭代性能与效率
· 2. 什么?你想跨数据库关联查询?
点击右上角即可分享
微信分享提示