不难- -! 哈夫曼树...

时间一粒 2010-06-08 06:03:10
编译没有错误!但是结果不是正确的!也就是哈夫曼树编码错误!
例:输入结点为3
权值为1 2 3
结果输出:
1(1):10
2(2):11
3(3):0
个人觉得是:
1(1):00
2(2):01
3(3):1

望高手指点!谢谢~
//根据给定的结点和权值,构建哈夫曼树,并进行哈夫曼编码
#include <stdio.h>
#include<malloc.h>
#include<string.h>
#include<stdlib.h>
#include<stdio.h>
#define OVERFLOW -2
#define OK 1

int m,s1,s2;
typedef struct
{
unsigned int weight;//权值
unsigned int parent,lchild,rchild;
}HTNode,*HuffmanTree; //动态分配数组存储哈夫曼树
typedef char *HuffmanCode; //动态分配数组存储哈夫曼编码表

void Select(HuffmanTree HT,int n)//在数组中寻找权值最小的两个结点
{
int i,j;
for(i = 1;i <= n;i++)
if(!HT[i].parent)
{
s1 = i;break;
}
for(j = i+1;j <= n;j++)
if(!HT[j].parent)
{
s2 = j;break;
}
for(i = 1;i <= n;i++)
{ if((HT[s1].weight>HT[i].weight)&&(!HT[i].parent)&&(s2!=i))
s1=i;}
for(j = 1;j <= n;j++)
{if((HT[s2].weight>HT[j].weight)&&(!HT[j].parent)&&(s1!=j))
s2=j;}
}

void HuffmanCoding(HuffmanTree &HT, HuffmanCode HC[], int *w, int n)
{ // 算法6.13
// w存放n个字符的权值(均>0),构造哈夫曼树HT,
// 并求出n个字符的哈夫曼编码HC
int i,j;
char *cd;
int p,cdlen;
if (n<=1) exit(OVERFLOW);//return ;
m = 2 * n - 1; //总节点个数
HT = (HuffmanTree)malloc((m+1) * sizeof(HTNode)); // 0号单元未用
for (i=1; i<=n; i++)
{ //初始化
HT[i].weight=w[i-1];
HT[i].parent=0;
HT[i].lchild=0;
HT[i].rchild=0;
}
for (i=n+1; i<=m; i++)
{ //初始化
HT[i].weight=0;
HT[i].parent=0;
HT[i].lchild=0;
HT[i].rchild=0;
}
puts("\n哈夫曼树的构造过程如下所示:");
printf("HT初态:\n 结点 weight parent lchild rchild");
for (i=1; i<=m; i++)
printf("\n%4d %8d %8d %8d %8d",i,HT[i].weight,HT[i].parent,HT[i].lchild, HT[i].rchild);

printf(" 按任意键,继续选择...");
getchar();
for (i=n+1; i<=m; i++)
{ // 建哈夫曼树
// 在HT[1..i-1]中选择parent为0且weight最小的两个结点,
// 其序号分别为s1和s2。
Select(HT, i-1);
HT[s1].parent = i; HT[s2].parent = i; //修改s1和s2结点的父指针parent
HT[i].lchild = s1; HT[i].rchild = s2; //修改i结点的左右孩子指针
HT[i].weight = HT[s1].weight + HT[s2].weight; //修改权值
printf("\nselect: s1=%d s2=%d\n", s1, s2);
printf(" 结点 weight parent lchild rchild");
for (j=1; j<=i; j++)
printf("\n%4d %8d %8d %8d %8d",j,HT[j].weight,
HT[j].parent,HT[j].lchild, HT[j].rchild);
printf(" 按任意键,继续选择...");
getchar();
}
//------无栈非递归遍历哈夫曼树,求哈夫曼编码
cd = (char *)malloc(n*sizeof(char)); //开辟一个求编码的工作空间
p = m; cdlen = 0;
for (i=1; i<=m; ++i) // 遍历哈夫曼树时用作结点状态标志
HT[i].weight = 0;
while (p)
{
if (HT[p].weight==0)
{ // 向左
HT[p].weight = 1;
if (HT[p].lchild != 0){p = HT[p].lchild; cd[cdlen++] ='0';} //若是左孩子编为'0'
else if (HT[p].rchild == 0)
{ // 登记叶子结点的字符的编码
HC[p] = (char *)malloc((cdlen+1) * sizeof(char)); //为第p个编码分配空间
cd[cdlen] ='\0';//编码结束符
strcpy(HC[p], cd); //将编码从cd复制到HC中
}
}
else if (HT[p].weight==1)
{ // 向右
HT[p].weight = 2;
if (HT[p].rchild != 0) { p = HT[p].rchild; cd[cdlen++] ='1'; }//若是右孩子编为'1'
}
else
{ // HT[p].weight==2,退回退到父结点,编码长度减1
HT[p].weight = 0;
p = HT[p].parent;
--cdlen;
}
}
} // HuffmanCoding

int main()
{
HuffmanTree HT;//哈夫曼树HT
HuffmanCode *HC;//哈夫曼编码表HC
int *w,n,i; //w存放叶子结点权值
printf("输入结点数:");//puts("输入结点数:");
scanf("%d",&n);
HC = (HuffmanCode *)malloc(n*sizeof(HuffmanCode)); //分配n个编码的头指针向量
w = (int *)malloc(n*sizeof(int));
printf("输入%d个结点的权值\n",n);
for(i = 0;i < n;i++)
{
scanf("%d",&w[i]);
}
HuffmanCoding(HT,HC,w,n);

puts("\n各结点的哈夫曼编码:");
for(i = 1;i <= n;i++)
printf("%2d(%2d):%s\n",i,w[i-1],HC[i]);
getchar();
}
...全文
224 7 打赏 收藏 转发到动态 举报
写回复
用AI写文章
7 条回复
切换为时间正序
请发表友善的回复…
发表回复
时间一粒 2010-06-09
  • 打赏
  • 举报
回复
[Quote=引用 5 楼 z569362161 的回复:]
void HuffmanCoding(HuffmanTree &HT, HuffmanCode HC[], int *w, int n)
[/Quote]
有问题?怎么错了?
z569362161 2010-06-09
  • 打赏
  • 举报
回复
void HuffmanCoding(HuffmanTree &HT, HuffmanCode HC[], int *w, int n)
时间一粒 2010-06-09
  • 打赏
  • 举报
回复
[Quote=引用 3 楼 wangyinbin 的回复:]
我刚刚做了课程设计,就是这个哈弗曼树
[/Quote]
麻烦那帮忙看看哪里出错了!
太乙 2010-06-09
  • 打赏
  • 举报
回复


lz看看这个http://baike.baidu.com/view/189694.htm

参考我以前写的一个可视化的哈弗曼树:

http://d.download.csdn.net/down/403519/hqin6
懒得搭理你 2010-06-08
  • 打赏
  • 举报
回复
我刚刚做了课程设计,就是这个哈弗曼树
时间一粒 2010-06-08
  • 打赏
  • 举报
回复
大家不要那么浮躁嘛!
静下心来,帮忙看看啊!
liutengfeigo 2010-06-08
  • 打赏
  • 举报
回复
OK,今天晚上我就看哈夫曼树

69,382

社区成员

发帖
与我相关
我的任务
社区描述
C语言相关问题讨论
社区管理员
  • C语言
  • 花神庙码农
  • 架构师李肯
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧