鱼C论坛

 找回密码
 立即注册
查看: 2353|回复: 0

[技术交流] 数据结构C语言版哈夫曼树

[复制链接]
发表于 2017-12-16 23:28:40 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#define OK 1
#define ERROR 0
#define OVERFLOW -1
#define n 5//叶子数目
#define m (2*n-1)//结点总数
#define maxval 10000
#define maxsize 100//哈夫曼编码的最大位数
typedef char TElemtype;
typedef int status;
typedef struct hufmtree
{
        float weight;
        int parent,lchild,rchild;
        char ch;//字符
}hufmtree;
typedef struct codetype
{
        char bits[n];//位串
        int start;//编码在位串中的开始位置
        char ch;//字符
}codetype;
void huffman(hufmtree tree[]);//建立哈夫曼树
void huffmancode(codetype code[],hufmtree tree[]);//根据哈夫曼树求出哈夫曼编码
void decode(hufmtree tree[]);//依次读入电文,根据哈夫曼树译码
int main()
{
        printf("哈夫曼编码:\n");
        hufmtree tree[m];
        codetype code[n];
        int i,j;//循环变量
        huffman(tree);//建立哈夫曼树
        huffmancode(code,tree);//根据哈夫曼树求出哈夫曼编码
        for(i=0;i<n;i++)
        {       
                for(j=code[i].start;j<n;j++)
                {
                        printf("\n");
                        printf("读入电文,并进行译码\n");
                        decode(tree);//依次读入电文,根据哈夫曼树译码
                }
       
        }
}
void huffman(hufmtree tree[])//建立哈夫曼树
{
        int i,j,p1,p2;//p1,p2分别记住每次合并时权值最小和次小的两个根结点的下标
        float small1,small2,f;
        char c;
        for(i=0;i<m;i++)//初始化
        {
                tree[i].parent=0;
                tree[i].lchild=-1;
                tree[i].rchild=-1;
                tree[i].weight=0.0;
        }
        printf("依次读入前%d个结点的字符及权值(中间用空格隔开)\n",n);
        for(i=0;i<n;i++)//读入前n个结点的字符及权值
        {
                printf("输入第%d个字符和权值:\n",i+1);
                scanf("%c %f",&c,&f);
                getchar();
                tree[i].ch=c;
                tree[i].weight=f;
        }
        for(i=n;i<m;i++)//进行n-1次合并,产生n-1个新结点
        {
                p1=0;p2=0;
                small1=maxval;small2=maxval;//maxval是float类型的最大值
                for(j=0;j<i;j++) //选出两个权值最小的根结点
                if(tree[j].parent==0)
                if(tree[j].weight<small1)
                {
                        small2=small1;//改变最小权、次小权及对应的位置
                        small1=tree[j].weight;
                        p2=p1;
                        p1=j;
                }
                else
                if(tree[j].weight<small2)
                {
                        small2=tree[j].weight;//改变次小权及位置
                        p2=j;
                }
                tree[p1].parent=i;
                tree[p2].parent=i;
                tree[i].lchild=p1;//最小权根结点是新结点的左孩子
                tree[i].rchild=p2;//次小权根结点是新结点的右孩子
                tree[i].weight=tree[p1].weight+tree[p2].weight;
        }
}
void huffmancode(codetype code[],hufmtree tree[])//根据哈夫曼树求出哈夫曼编码

{
        int i,c,p;
        codetype cd;//缓冲变量
        for(i=0;i<n;i++)
        {
                cd.start=n;
                cd.ch=tree[i].ch;
                c=i;//从叶结点出发向上回溯
                p=tree[i].parent;//tree[p]是tree[i]的双亲
                while(p!=0)
                {
                        cd.start--;
                        if(tree[p].lchild==c)
                        cd.bits[cd.start]='0'; //tree[i]是左子树,生成代码'0'
                        else
                        cd.bits[cd.start]='1'; //tree[i]是右子树,生成代码'1'
                        c=p;
                        p=tree[p].parent;
                }
                code[i]=cd;//第i+1个字符的编码存入code[i]
        }
}
void decode(hufmtree tree[])//依次读入电文,根据哈夫曼树译码
{
        int i,j=0;
        char b[maxsize];
        char endflag='2';//电文结束标志取2
        i=m-1;//从根结点开始往下搜索
        printf("输入发送的编码(以'2'为结束标志):");
        gets(b);
        printf("译码后的字符为");
        while(b[j]!='2')
        {
                if(b[j]=='0')
                i=tree[i].lchild;  //走向左孩子
                else
                i=tree[i].rchild;//走向右孩子
                if(tree[i].lchild==-1)//tree[i]是叶结点
                {
                        printf("%c",tree[i].ch);
                        i=m-1; //回到根结点
                 }
                 j++;
        }
        printf("\n");
        if(tree[i].lchild!=-1&&b[j]!='2')//电文读完,但尚未到叶子结点
        printf("ERROR\n");//输入电文有错
}
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-19 08:06

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表