标题:急,求哈夫曼编码
只看楼主
danluck
Rank: 1
等 级:新手上路
帖 子:1
专家分:0
注 册:2007-6-23
 问题点数:0 回复次数:2 
急,求哈夫曼编码
问题描述:利用哈夫曼编码进行通信可以大大提高信道利用率,缩短信息传输时间,降低传输成本。但是,这要求在发送端通过一个编码系统对待传数据预先编码,在接收端将传来的数据进行译码(复原)。对于双工信道(即可以双向传输信息的信道),每端都需要一个完整的编/译码系统。试为这样的信息收发站写一个哈夫曼码的编/译码系统。
基本要求:
1、 权值数据存放在数据文件(文件名为data.txt,位于执行程序的当前目录中)
2、 采用静态存储结构;
3、 初始化:键盘输入字符集大小n、n个字符和n个权值,建立哈夫曼树;
4、 编码:利用建好的哈夫曼树生成哈夫曼编码;
5、 输出编码;
6、 设字符集及频度如下表:
字符 空格 A B C D E F G H I J K L M
频度 186 64 13 22 32 103 21 15 47 57 1 5 32 20
字符 N O P Q R S T U V W X Y Z
频度 57 63 15 1 48 51 80 23 8 18 1 16 1

请各位高手帮忙,小女子不胜感激!
搜索更多相关主题的帖子: 哈夫曼编码 
2007-06-23 10:57
zhufangzeng
Rank: 1
等 级:新手上路
帖 子:1
专家分:0
注 册:2008-10-25
得分:0 
下面的程序可能对你有用
java程序

//构造哈夫曼树并获得哈夫曼编码

class HaffmanNode                                //哈夫曼树的结点类
{
    int weight;                                  //权值
    int parent,left,right;                       //父母结点和左右孩子下标
   
    public HaffmanNode(int weight)
    {                    
        this.weight = weight;
        this.parent=-1;
        this.left=-1;
        this.right=-1;
    }
    public HaffmanNode()
    {                    
        this(0);
    }
    public String toString()
    {
        return this.weight+", "+this.parent+", "+this.left+", "+this.right;
    }
}

public class HaffmanTree                         //哈夫曼树
{
    private int leafNum;                         //叶子结点个数
    private HaffmanNode[] hnodes;                //哈夫曼树的结点数组
        
    public HaffmanTree(int[] weight)             //构造指定权值集合的哈夫曼树
    {
        int n = weight.length;                   //n个叶子结点
        this.leafNum = n;
        this.hnodes = new HaffmanNode[2*n-1];    //n个叶子结点的哈夫曼树共有2n-1个结点
        for(int i=0; i<n; i++)                   //结点数组初始化有n个叶子结点
            this.hnodes[i] = new HaffmanNode(weight[i]);


        for(int i=0; i<n-1; i++)                 //构造n-1个2度结点,每循环一次,构造一个2度结点
        {
            int min1, min2, x1, x2;
            min1 = min2 = Integer.MAX_VALUE;     //选择最小和次最小权值,初值为最大权值
            x1 = x2 = -1;                        //记录两个无父母的最小权值结点下标
            for(int j=0; j<n+i; j++)             //查找两个无父母的最小权值结点
            {
                if(hnodes[j].weight<min1 && hnodes[j].parent==-1)
                {
                    min2 = min1;
                    x2 = x1;
                    min1 = hnodes[j].weight;     //min1记下最小权值
                    x1 = j;                      //x1记下最小权值结点的下标
                }
                else if(hnodes[j].weight<min2 && hnodes[j].parent==-1)
                {
                    min2 = hnodes[j].weight;
                    x2 = j;                      //x2记下次最小权值结点的下标
                }
            }

            hnodes[x1].parent  = n+i;            //将找出的两棵权值最小的子树合并为一棵子树
            hnodes[x2].parent  = n+i;
            this.hnodes[n+i] = new HaffmanNode();
            hnodes[n+i].weight = hnodes[x1].weight+hnodes[x2].weight;
            hnodes[n+i].left = x1;
            hnodes[n+i].right = x2;
        }
    }

    public String toString()
    {
        String str="";
        for (int i=0; i<this.hnodes.length && hnodes[i]!=null; i++)
            str += i+", "+this.hnodes[i].toString()+"\n";
        return str;
    }

    public String[] haffmanCode()                //返回当前哈夫曼树的哈夫曼编码
    {
        String[] code = new String[this.leafNum];
        for(int i=0; i<this.leafNum; i++)        //求n个叶子结点的哈夫曼编码
        {
            code[i]="";
            int child = i;
            int parent = hnodes[child].parent;
            while (parent!=-1)                   //由叶结点向上直到根结点循环
            {
                if(hnodes[parent].left==child)
                    code[i]="0"+code[i];         //左孩子结点编码为0
                else
                    code[i]="1"+code[i];         //右孩子结点编码为1
                child = parent;
                parent = hnodes[child].parent;        
            }
        }
        return code;
    }

    public static void main(String[] args)
    {
        int[] weight={186 64 13 22 32 103 21 15 47 57 1 5 32 20};      //指定权值集合
        HaffmanTree htree = new HaffmanTree(weight);
        System.out.println("哈夫曼树的结点数组:\n"+htree.toString());
        String[] code = htree.haffmanCode();
        System.out.println("哈夫曼编码:");
        for (int i=0; i<code.length; i++)
             System.out.println(code[i]);
    }
}
/*
将各个字母出现的频度照如下输入int[] weight={186 64 13 22 32 103 21 15 47 57 1 5 32 20};求出各个频度的编码后,你再对照着还原成字符的编码吧。
2008-11-20 20:50
twlkyao
该用户已被删除
得分:0 
提示: 作者被禁止或删除 内容自动屏蔽
2010-05-09 23:21



参与讨论请移步原网站贴子:https://bbs.bccn.net/thread-149667-1-1.html




关于我们 | 广告合作 | 编程中国 | 清除Cookies | TOP | 手机版

编程中国 版权所有,并保留所有权利。
Powered by Discuz, Processed in 0.191884 second(s), 7 queries.
Copyright©2004-2025, BCCN.NET, All Rights Reserved