#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#define N 100
#define M 2*N-1
typedef char *HuffmanCode[2*M];
typedef struct
{
 int weight;//权值
 int parent;//父节点
 int Lchild;//左
 int Rchild;//右
}HTNode,Huffman[M+1];//huffman树
typedef struct Node
{
 int weight;//叶子权值
 char c;//叶子
 int num;//叶子的2进制码长度
}WNode,WeightNode[N];
void CreatWeight(char ch[],int *s,WeightNode CW,int *p)//生成叶子节点字符与权值
{
 int i,j,k;
 int tag;
 *p=0;//叶子节点数
 for(i=0;ch[i]!='\0';i++)//记录字符个数放入CW
 {
  tag=1;
  for(j=0;j<i;j++)
   if(ch[j]==ch[i])
   {
    tag=0;
    break;
   }
   if(tag)
   {
    CW[++*p].c=ch[i];
    CW[*p].weight=1;
    for(k=i+1;ch[k]!='\0';k++)
    {
     if(ch[i]==ch[k])
      CW[*p].weight++;//权值++
    }
   }
 }
 *s=i;//字符串长
}
void CreateHuffmanTree(Huffman ht,WeightNode w,int n)//建立huffmantree
{
 int i,j;
 int s1,s2;
 for(i=1;i<=n;i++)//初始化
 {
  ht[i].weight=w[i].weight;
  ht[i].parent=0;
  ht[i].Lchild=0;
  ht[i].Rchild=0;
 }
 for(i=n+1;i<=2*n-1;i++)
 {
  ht[i].weight=0;
  ht[i].parent=0;
  ht[i].Lchild=0;
  ht[i].Rchild=0;
 }
 for(i=n+1;i<=2*n-1;i++)
 {
  for(j=1;j<=i-1;j++)
   if(!ht[j].parent)
    break;
   s1=j;//找第一个双亲!=0节点
   for(;j<=i-1;j++)
    if(!ht[j].parent)
     s1=ht[s1].weight>ht[j].weight?j:s1;
    ht[s1].parent=i;
    ht[i].Lchild=s1;
    for(j=1;j<=i-1;j++)
     if(!ht[j].parent)
      break;
     s2=j;//找第二个双亲!=0节点
     for(;j<=i-1;j++)
      if(!ht[j].parent)
       s2=ht[s2].weight>ht[j].weight?j:s2;
    //  printf("s2=%d\n",s2);
      ht[s2].parent=i;
      ht[i].Rchild=s2;
               //  intf("rchild=%d\n",ht[i].Rchild);
      ht[i].weight=ht[s1].weight+ht[s2].weight;//权值++
 }
}
void CrtHuffmanNodeCode(Huffman ht,char ch[],HuffmanCode h,WeightNode weight,int m,int n)//叶子节点编码
{
 int i,c,p,start;
 char *cd;
 cd=(char*)malloc(n*sizeof(char));
 cd[n-1]='\0';//末尾=0
 for(i=1;i<=n;i++)
 {
  start=n-1;//cd从末尾开始
  c=i;
  p=ht[i].parent;//p在n+1——2n-1
  while(p)//沿父方向遍历至0
  {
   start--;//向前
   if(ht[p].Lchild==c)//与lchild相同=0
    cd[start]='0';
   else//=1
    cd[start]='1';
   c=p;
   p=ht[p].parent;
  }
  weight[i].num=n-start;//二进制码长度
  h[i]=(char*)malloc((n-start)*sizeof(char));
  strcpy(h[i],&cd[start]);//二进制字符复制到h
 }
 free(cd);
 //system("pause");
}
void CrtHuffmanCode(char ch[],HuffmanCode h,HuffmanCode hc,WeightNode weight,int n,int m)//所有字符编码
{
 int i,k;
 for(i=0;i<m;i++)
 {
  for(k=1;k<=n;k++)//从weight[k].c中找与ch[i]相等下标k*
   if(ch[i]==weight[k].c)
    break;
   hc[i]=(char*)malloc((weight[k].num)*sizeof(char));
   strcpy(hc[i],h[k]);//复制二进制码
 }
}
void TrsHuffmanTree(Huffman ht,WeightNode w,HuffmanCode hc,int n,int m)//译码
{
 int i=0,j,p;
 while(i<m)
 {
  p=2*n-1;//从父亲遍历到叶子
  for(j=0;hc[i][j]!='\0';j++)
  {
   if(hc[i][j]=='0')
    p=ht[p].Lchild;
   else
    p=ht[p].Rchild;
  }
  printf("%c",w[p].c);
  i++;
 }
}
void main()
{
 int i;
 int n=0;//叶子数
 int m=0;//ch长度
 char ch[N];//输入字符
 Huffman ht;//huffman二叉树
 HuffmanCode h;//叶子编码
 HuffmanCode hc;//所有节点编码
 WeightNode weight;//叶子信息
 printf("Huffman\n  please intput inf\n");
 gets(ch);
 CreatWeight(ch,&m,weight,&n);
 for(i=1;i<=n;i++)//输出叶子字符和权
  printf("%c",weight[i].c);
 printf("\nweight");
 for(i=1;i<=n;i++)
  printf("%d",weight[i].weight);
 CreateHuffmanTree(ht,weight,n);
 printf("\ti\tweight\tparent\tLChild\tRChild\n");
 for(i=1;i<=2*n-1;i++)
  printf("\t%d\t%d\t%d\t%d\t%d\n",i,ht[i].weight,ht[i].parent,ht[i].Lchild,ht[i].Rchild);
 CrtHuffmanNodeCode(ht,ch,h,weight,m,n);
 for(i=1;i<=n;i++)
 {
  printf("\t%c ",weight[i].c);
  printf("%s\n",h[i]);
 }
 CrtHuffmanCode(ch,h,hc,weight,n,m);
 printf("code");
 for(i=0;i<m;i++)
  printf("%s",hc[i]);
 printf("\n");
 TrsHuffmanTree(ht,weight,hc,n,m);
}

haffman树c实现的更多相关文章

  1. Haffman编码(haffman树)

    Haffman编码 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 哈弗曼编码大家一定很熟悉吧(不熟悉也没关系,自己查去...).现在给你一串字符以及它们所对应的权值 ...

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

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

  3. java实现Haffman编码

    1.先创建一个树节点类(泛型类),为了方便使用集合的排序方法,泛型类要实现泛型接口Comparable,代码如下 package com.hjp.huffman; /** * Created by J ...

  4. 文件压缩小项目haffman压缩

    文件压缩的原理: 文件压缩总体可以分为有损压缩和无损压缩两类,有损压缩是指对mp3等格式的文件,忽略一些无关紧要的信息,只保留一些关键的信息,但并不因此影响用户对于这些mp3格式文件的体验度,无损压缩 ...

  5. Huffman Tree

    哈夫曼(Huffman)树又称最优二叉树.它是一种带权路径长度最短的树,应用非常广泛. 关于Huffman Tree会涉及到下面的一些概念: 1. 路径和路径长度路径是指在树中从一个结点到另一个结点所 ...

  6. ZJOI2019一轮停课刷题记录

    Preface 菜鸡HL终于狗来了他的省选停课,这次的时间很长,暂定停到一试结束,不过有机会二试的话还是可以搞到4月了 这段时间的学习就变得量大而且杂了,一般以刷薄弱的知识点和补一些新的奇怪技巧为主. ...

  7. [数据结构] 大纲 - Stan Zhang 数据结构速通教程

    * 注: 本文/本系列谢绝转载,如有转载,本人有权利追究相应责任. 2019年4月8日 P1.1 链表 Link:https://www.cnblogs.com/yosql473/p/10727471 ...

  8. JPEG编码(二)

    来自CSDN评论区http://bbs.csdn.net/topics/190980 1. 色彩模型 JPEG 的图片使用的是 YCrCb 颜色模型, 而不是计算机上最常用的 RGB. 关于色彩模型, ...

  9. c++实验8 哈夫曼编码-译码器

    哈夫曼编码-译码器 此次实验的注释解析多加不少---若对小伙伴们有帮助 希望各位麻烦点个关注 多谢 1.哈夫曼树构造算法为: (1)由给定的n个权值{w1,w2,…,wn}构造n棵只有根结点的二叉树, ...

随机推荐

  1. SlidingMenu侧换菜单的导入

    对于Adt-22.3有一种使用SlidingMenu(侧滑菜单的方式),直接加你放到lib文件夹下

  2. Android TV 模拟器启动

    模拟器启动错误 使用IntelCPU的模拟器 http://software.intel.com/en-us/android/articles/intel-hardware-accelerated-e ...

  3. OpenCart本地调试环境搭建

    OpenCart简介: 免费开源网络版电子商务系统,是建立在线商务网站首选之一.有众多用户和开发基础,结合其丰富特性与模板插件,可最大化定制在线商店.(也就是用来方便开网店的) 本地调试准备: Fir ...

  4. Java设计模式(学习整理)----装饰模式

    1.概念: (在我看来,模式就像是是一种思想,在这种思想的指引下,对代码和结构的一番加工和整合而已!都是套路!) 装饰模式又称包装(Wrapper)模式,是以对客户端透明的方式扩展对象的功能,是继承关 ...

  5. 你好,C++(3)2.1 一个C++程序的自白

    第2部分 与C++第一次亲密接触 在浏览了C++“三分天下”的世界版图之后,便对C++有了基本的了解,算是一只脚跨入了C++世界的大门.那么,怎样将我们的另外一只脚也跨入C++世界的大门呢?是该即刻开 ...

  6. 【vc】1_Windows程序内部运行机制

    创建一个Win32应用程序步骤: 1.编写WinMain函数; 2.创建窗口(步骤如下): a.设计(一个)窗口类(WNDCLASS) b.注册(该)窗口类. c.创建窗口. d.显示并更新窗口. 3 ...

  7. PHP: 使用CURL访问FTP

    今天要做FTP上传.本想用PHP自带的FTP函数来实现,结果发现这个模块没有编译进来,重新编译PHP太麻烦,改用其他方式实现吧  FTP上传 if (isset($_POST['Submit'])) ...

  8. Android自定义View基础

    自定义控件, 视频教程 http://www.jikexueyuan.com/course/1748.html 1. 编写自定义view 2. 加入逻辑线程 3. 提取和封装自定义view 4. 利用 ...

  9. C#.NET Winform 通用开发框架

    C/S系统开发框架-企业版 V4.0 (Enterprise Edition) 简介: http://www.csframework.com/cs-framework-4.0.htm 视频下载: 百度 ...

  10. WebApi Gzip(Deflate) 压缩请求数据

    由于不能直接访问指定数据库,只能通过跳板机查询Oracle数据,所以要做一个数据中转接口, 查询数据就要压缩,于是就找资料,代码如下,其中要注意的是Response.Headers.Remove(&q ...