<?php
namespace Test; use Iterator;
use ArrayAccess;
use Exception; // 叶子结点或者连接结点的基类
class HuffmanBase
{
protected $weight; // 权重
protected $parent; public function setParent($parent)
{
$this->parent = $parent;
} public function getWeight()
{
return $this->weight;
} public function getParent()
{
return $this->parent;
}
} // 叶子结点
class HuffmanLeafNode extends HuffmanBase
{
protected $code; // 需要编码的字母 public function __construct($weight,$code,$parent = null)
{
if(!is_int($weight) || $weight <= 0 || is_null($code))
{
throw new Exception('Param is error!' .__FILE__.__LINE__);
}
$this->weight = abs($weight);
$this->code = $code;
$this->parent = $parent;
} public function getCode()
{
return $this->code;
}
} // 连接结点
class HuffmanJoinNode extends HuffmanBase
{
protected $lChild;
protected $rChild; public function __construct($weight = 0,$lChild = null,$rChild = null)
{
$this->weight = $weight;
$this->rChild = $rChild;
$this->lChild = $lChild;
} public function setWeight($leftWeight,$rightWeight)
{
if(!is_int($this->rChild) || !is_int($this->lChild))
{
throw new \Exception("Please initialize the left child or the right child!\n");
}
$this->weight = $leftWeight + $rightWeight;
} public function setChild($child,$leftOrRight)
{
if('left' == $leftOrRight)
{
$this->lChild = $child;
}
elseif('right' == $leftOrRight)
{
$this->rChild = $child;
}
else
{
throw new \Exception("Please input 'left' or 'right' to leftOrRight!\n");
}
} public function getChild($leftOrRight)
{
if('left' == $leftOrRight)
{
return $this->lChild;
}
elseif('right' == $leftOrRight)
{
return $this->rChild;
}
else
{
throw new \Exception("Please input 'left' or 'right' to leftOrRight!\n");
}
}
} // 哈夫曼树
class HuffmanTree implements \ArrayAccess,\Iterator
{
protected $nodes = array(); public function &getAllNodes()
{
return $this->nodes;
} public function offsetExists($offset)
{
// TODO: Implement offsetExists() method.
return isset($this->nodes[$offset]);
} public function offsetGet($offset)
{
// TODO: Implement offsetGet() method.
if(isset($this->nodes[$offset]))
{
return $this->nodes[$offset];
}
else
{
return null;
}
} public function offsetSet($offset,$value)
{
// TODO: Implement offsetSet() method.
if(!($value instanceof HuffmanBase))
{
throw new Exception('Param is error!' .__FILE__.__LINE__);
} if(is_null($offset))
{
$this->nodes[] = $value;
}
else
{
$this->nodes[$offset] = $value;
}
} public function offsetUnset($offset)
{
// TODO: Implement offsetUnset() method.
unset($this->nodes[$offset]);
} public function current()
{
// TODO: Implement current() method.
return current($this->nodes);
} public function key()
{
// TODO: Implement key() method.
return key($this->nodes);
} public function next()
{
// TODO: Implement next() method.
next($this->nodes);
} public function rewind()
{
// TODO: Implement rewind() method.
reset($this->nodes);
} public function valid()
{
// TODO: Implement valid() method.
return $this->offsetExists(key($this->nodes));
} public function length()
{
return count($this->nodes);
}
} // 从[$left,$right]区间选择parent=0并且weight最小的两个结点,其序号分别为$minNode1,$minNode2;
function selectTwoMinWeightNode(HuffmanTree &$huffmanTree,$left,$right,&$minNode1,&$minNode2)
{
$left = abs($left);
$right = abs($right); if(!is_int($left) || !is_int($right) || $left == $right)
{
throw new Exception('Param is error!' .__FILE__.__LINE__);
} if($left > $right)
{
$tmp = $left;
$left = $right;
$right = $tmp;
} $nodes = $huffmanTree->getAllNodes();
if(!isset($nodes[$right]))
{
throw new Exception('Over the array index!'.__FILE__.__LINE__);
} $tmp = array();
for($i = $left;$i <= $right; ++$i)
{
$huffmanNode = $huffmanTree[$i];
if(!is_null($huffmanNode->getParent()))
{
continue;
}
$tmp[$i] = $huffmanNode->getWeight();
} if(count($tmp) <= 1)
{
throw new Exception('Not enough number!'.__FILE__.__LINE__);
}
asort($tmp,SORT_NUMERIC);
$t = array_keys($tmp);
$minNode1 = $t[0];
$minNode2 = $t[1];
} // (编码 => 权重)
$nodes = array('A' => 3, 'B' => 4, 'C' => 7, 'D' => 10); $huffmanTree = new HuffmanTree(); // 初始化哈夫曼树的叶子结点
foreach($nodes as $code => $weight)
{
$huffmanNode = new HuffmanLeafNode($weight,$code);
$huffmanTree[] = $huffmanNode;
} $leafCount = $huffmanTree->length(); // 叶子结点的数量(大于1的值)
$nodeCount = 2 * $leafCount -1 ; // 哈夫曼树结点的数量 // 初始化哈夫曼树的非叶子结点(如果编译器未优化,--$i应该是比$i++效率高点的)
for($i = $nodeCount - $leafCount;$i >= 1; --$i)
{
$huffmanNode = new HuffmanJoinNode();
$huffmanTree[] = $huffmanNode;
} // 建立哈夫曼树
for($i = $leafCount;$i < $nodeCount; ++$i)
{
selectTwoMinWeightNode($huffmanTree,0,$i-1,$minNode1,$minNode2);
$huffmanNode1 = $huffmanTree[$minNode1];
$huffmanNode2 = $huffmanTree[$minNode2];
$huffmanNode1->setParent($i);
$huffmanNode2->setParent($i);
$huffmanTree[$i]->setChild($minNode1,'left');
$huffmanTree[$i]->setChild($minNode2,'right');
$huffmanTree[$i]->setWeight($huffmanNode1->getWeight(),$huffmanNode2->getWeight());
} // 从叶子到根的遍历,得到字母的编码
$huffmanCode = array();
for($i = 0;$i < $leafCount; ++$i)
{
$leafNode = $huffmanTree[$i];
$code = $leafNode->getCode();
$reverseCode = array();
for($c = $i,$pi = $leafNode->getParent();!is_null($pi);$pi = $huffmanTree[$pi]->getParent())
{
$huffmanNode = $huffmanTree[$pi];
if($huffmanNode->getChild('left') === $c)
{
$reverseCode[] = 0;
}
elseif($huffmanNode->getChild('right') === $c)
{
$reverseCode[] = 1;
}
else
{
throw new Exception('Something error happened!' .__FILE__.__LINE__);
}
$c = $pi;
}
$huffmanCode[$code] = array_reverse($reverseCode);
} foreach($huffmanCode as $key => $value)
{
$s = implode(',',$value);
echo $key. " : " .$s ."\n";
}

运行结果:

运行环境:

ArrayAccess  :(PHP 5 >= 5.0.0, PHP 7)

PHP 哈夫曼的实现的更多相关文章

  1. 哈夫曼(huffman)树和哈夫曼编码

    哈夫曼树 哈夫曼树也叫最优二叉树(哈夫曼树) 问题:什么是哈夫曼树? 例:将学生的百分制成绩转换为五分制成绩:≥90 分: A,80-89分: B,70-79分: C,60-69分: D,<60 ...

  2. (哈夫曼树)HuffmanTree的java实现

    参考自:http://blog.csdn.net/jdhanhua/article/details/6621026 哈夫曼树 哈夫曼树(霍夫曼树)又称为最优树. 1.路径和路径长度在一棵树中,从一个结 ...

  3. 数据结构之C语言实现哈夫曼树

    1.基本概念 a.路径和路径长度 若在一棵树中存在着一个结点序列 k1,k2,……,kj, 使得 ki是ki+1 的双亲(1<=i<j),则称此结点序列是从 k1 到 kj 的路径. 从 ...

  4. (转载)哈夫曼编码(Huffman)

    转载自:click here 1.哈夫曼编码的起源: 哈夫曼编码是 1952 年由 David A. Huffman 提出的一种无损数据压缩的编码算法.哈夫曼编码先统计出每种字母在字符串里出现的频率, ...

  5. C++哈夫曼树编码和译码的实现

    一.背景介绍: 给定n个权值作为n个叶子结点,构造一棵二叉树,若带权路径长度达到最小,称这样的二叉树为最优二叉树,也称为哈夫曼树(Huffman Tree).哈夫曼树是带权路径长度最短的树,权值较大的 ...

  6. 数据结构图文解析之:哈夫曼树与哈夫曼编码详解及C++模板实现

    0. 数据结构图文解析系列 数据结构系列文章 数据结构图文解析之:数组.单链表.双链表介绍及C++模板实现 数据结构图文解析之:栈的简介及C++模板实现 数据结构图文解析之:队列详解与C++模板实现 ...

  7. HDU2527 哈夫曼编码

    Safe Or Unsafe Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  8. *HDU1053 哈夫曼编码

    Entropy Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Sub ...

  9. YTU 3027: 哈夫曼编码

    原文链接:https://www.dreamwings.cn/ytu3027/2899.html 3027: 哈夫曼编码 时间限制: 1 Sec  内存限制: 128 MB 提交: 2  解决: 2 ...

  10. 哈夫曼树---POJ3253

    http://poj.org/problem?id=3253 这就是 最典型的哈夫曼树的题型,我们就根据这道题学习一下哈夫曼树 这是最开始我们把21据下来之后我们据下8,然后再据下5得到34,可以看出 ...

随机推荐

  1. Metasploit工具----漏洞利用模块

    漏洞利用是指由渗透测试者利用一个系统.应用或者服务中的安全漏洞进行的攻击行为.流行的渗透攻击技术包括缓冲区溢出.Web应用程序攻击,以及利用配置错误等,其中包含攻击者或测试人员针对系统中的漏洞而设计的 ...

  2. 基于python-django框架的支付宝支付案例

    目录 @ 一. 开发前的准备 1. 必须了解的知识 SDK:软件开发工具包,可以为开发者提供快速开发的工具 沙箱环境:也就是测试环境 支付宝支付金额的精度:小数点后两位(面试) 支付宝用的什么加密方式 ...

  3. 使用gdb调试c++程序

    上篇(使用c++开发跨平台程序)说到,我不怕造东西,我怕的是造出来的东西,如果出了问题,我却不知道原因.所以调试分析是一个重要的手段. C++调试是一个复杂的活.虽然大部分调试可以通过IDE在开发期间 ...

  4. ##发送post时,设置了utf-8,中文还是乱码?

    发送post时,设置了utf-8,中文还是乱码? 我们用HttpUrlConnection或HttpClient发送了post请求,其中有中文,虽然我们两边都设置了utf-8,但还是乱码? 我们在re ...

  5. filebeat相关registry文件内容解析

    filebeat的registry文件中存放的是被采集的所有日志的相关信息. linux中registry中一条日志记录的内容如下 {"source":"/var/log ...

  6. SpringBootSecurity学习(02)网页版登陆配置类代替默认配置

    增加Security配置类 前面演示了一个简单的登录入门例子,使用springboot-security默认的配置实现,虽然非常简单,但是基本实现了登录功能.不过在生产环境下,显然不能仅仅使用如此简单 ...

  7. C#深入学习笔记 - 可空类型与构造函数默认参数

    在实际开发中或许可能会遇到某个属性需要提供一个默认参数,如果该参数是引用类型的话,可以通过 使用 null 来表示未知的值,但如果是int或 其他值类型的话就有点不好办了,因为如果需要一个int或fl ...

  8. Java中Synchronized的优化原理

    我们知道,从 JDK1.6 开始,Java 对 Synchronized 同步锁做了充分的优化,甚至在某些场景下,它的性能已经超越了 Lock 同步锁.那么就让我们来看看,它究竟是如何优化的. 原本的 ...

  9. Docker5-docker私库的搭建及常用方法-harbor-registry方式

    一.简介 1.官方已经提供registry镜像为什么还需要用harbor 1)registry缺少镜像清理机制,可以push但是不能删除,耗费空间 2)registry缺乏相应的扩展机制 3)harb ...

  10. js实现烟花效果

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...