哈夫曼树

定义:给定n个带有权值的叶节点,将其组成一颗带权路径长度最小的二叉树,则该二叉树为哈夫曼树,亦称最优二叉树

假设叶节点的权值分别为w1, w2, w3 ... wn,到根节点的路径长度分别为l1, l2, l3 ... ln,

那么WPL = w1 * l1 + w2 * l2 + w3 * l3 + ... + wn * ln,哈夫曼树则是WPL最小的树

 

构成哈夫曼树算法

 

样例:

 

 

简单示例代码:

 

// Author: Waihui Zheng
// 哈夫曼树

#include <iostream>
#include <string>
#include <vector>
#include <queue>
#include <functional>

namespace my_space {

#define ENDTREE -1

class HaffMan {
    // 哈夫曼树节点
    struct TreeNode {
        int weight;
        int parent;
        int lchild;
        int rchild;

        TreeNode(int w = 0) : weight(w), parent(ENDTREE), lchild(ENDTREE), rchild(ENDTREE) {}
    };

    // 信息节点
    struct NodeInfo {
        std::string info;
        int weight;
        int location;

        NodeInfo(const std::string& str = "", int w = 0)
                : info(str), weight(w), location(ENDTREE) {}
    };

    // 优先队列节点
    struct WeightHelper {
        int weight;
        int index;

        WeightHelper(int w = 0, int ix = -1) : weight(w), index(ix) {}
        bool operator>(const WeightHelper& rhs) const {
            return weight > rhs.weight;
        }
    };

public:
    HaffMan() {}
    ~HaffMan() {}

    void clear() {
        _node_infos.clear();
        _haffman_tree.clear();
    }

    void add_info(const std::string& info, int weight) {
        _node_infos.push_back(NodeInfo(info, weight));
    }

    void construct_haffman_tree() {
        // 初始化哈夫曼树,建立n个独立的树
        _haffman_tree.clear();
        for (int i = 0; i < _node_infos.size(); ++i) {
            _haffman_tree.push_back(TreeNode(_node_infos[i].weight));
            _node_infos[i].location = i;
        }

        // 优先队列,内部按权值进行排序
        std::priority_queue<WeightHelper, std::vector<WeightHelper>, std::greater<WeightHelper> > pq;
        for (int i = 0; i < _haffman_tree.size(); ++i) {
            pq.push(WeightHelper(_haffman_tree[i].weight, i));
        }

        // 生成n-1颗树
        int times = _node_infos.size() - 1;
        while (times--) {
            // 优先找出两颗根权值最小的两颗
            WeightHelper lchild = pq.top();
            pq.pop();
            WeightHelper rchild = pq.top();
            pq.pop();

            // 创建新root节点
            TreeNode new_tree_node(
                    _haffman_tree[lchild.index].weight + _haffman_tree[rchild.index].weight);
            new_tree_node.lchild = lchild.index;
            new_tree_node.rchild = rchild.index;
            _haffman_tree.push_back(new_tree_node);

            // 修改子节点父母节点指针指向
            _haffman_tree[lchild.index].parent = _haffman_tree.size() - 1;
            _haffman_tree[rchild.index].parent = _haffman_tree.size() - 1;

            // 加入新树
            pq.push(WeightHelper(new_tree_node.weight, _haffman_tree.size() - 1));
        }
    }

    std::string get_haffman_code(const std::string& name) {
        int name_index = 0;
        for (; name_index < _node_infos.size(); ++name_index) {
            if (_node_infos[name_index].info == name) {
                break;
            }
        }

        std::string code;
        if (name_index == _node_infos.size()) {
            return code;
        }

        int cur_index = _node_infos[name_index].location;
        int parent_index = _haffman_tree[cur_index].parent;
        // 回溯寻找 叶子节点到根节点的路径
        while (parent_index != ENDTREE) {
            code.push_back(_haffman_tree[parent_index].lchild == cur_index ? '0' : '1');

            cur_index = parent_index;
            parent_index = _haffman_tree[cur_index].parent;
        }

        std::reverse(code.begin(), code.end());
        return code;
    }

    void print() {
        for (int i = 0; i < _node_infos.size(); ++i) {
            std::cout << _node_infos[i].info << " code: "
                      << get_haffman_code(_node_infos[i].info) << std::endl;
        }
    }

private:
    std::vector<NodeInfo> _node_infos;
    std::vector<TreeNode> _haffman_tree;
};

}

int main() {
    my_space::HaffMan haffman_tree;
    haffman_tree.add_info("A", 7);
    haffman_tree.add_info("B", 5);
    haffman_tree.add_info("C", 2);
    haffman_tree.add_info("D", 4);

    haffman_tree.construct_haffman_tree();

    haffman_tree.print();
    return 0;
}

 

posted on 2016-02-18 19:07  bug睡的略爽  阅读(268)  评论(0编辑  收藏  举报

导航