Huffman.c
Upload User: shenhu72
Upload Date: 2021-11-20
Package Size: 126k
Code Size: 10k
Development Platform:

Visual C++

  1. /***********************************************************************************************************
  2. Huffman.c
  3. 本演示程序提供了哈夫曼编码法的压缩和解压缩函数,并实现了对图象
  4. 文件的压缩和解压缩
  5. **********************************************************************************************************/
  6. #include <stdio.h>
  7. #include <stdlib.h>
  8. #include <string.h>
  9. /* 函数原型 */
  10. int  Huffman_Compression(char * infilename, char * outfilename);
  11. int  Huffman_Decompression(char * infilename, char * outfilename);
  12. /* 内部函数 */
  13. unsigned short  generate_code_table ();
  14. void          build_code_tree ();
  15. void  build_initial_heap ();
  16. void          compress_image ();
  17. void          get_frequency_count ();
  18. void build_decomp_tree ();
  19. void decompress_image ();
  20. /* 全局变量 */
  21. short           father[512];
  22. unsigned short  code[256], heap_length;
  23. unsigned long   compress_charcount, file_size, heap[257];
  24. unsigned char   code_length[256];
  25. long            frequency_count[512];
  26. short           decomp_tree[512];
  27. FILE            *ifile, *ofile;
  28. /* 主程序 */
  29. void main(int argc, char *argv[])
  30. {
  31. printf("Huffman compression and decompression utilityn");
  32. if (4 != argc) 
  33. {
  34. printf("nUsage : huffman -c|d sourcefilename targetfilenamen");
  35. exit(0);
  36. }
  37. if (! strcmp(argv[1], "-c"))
  38. {
  39. printf("nCompress...");
  40. Huffman_Compression(argv[2], argv[3]);
  41. }
  42. else if (! strcmp(argv[1], "-d"))
  43. {
  44. printf("nDecompress...");
  45. Huffman_Decompression(argv[2], argv[3]);
  46. }
  47. else
  48. printf("nUnknow command.n");
  49. }
  50. /*********************************************************************************
  51.   Huffman_Compression ()
  52.   用Huffman编码压缩指定的文件,并将结果存为新的文件
  53.  ********************************************************************************/
  54. int Huffman_Compression(char * infilename, char * outfilename)
  55. {
  56. if ((ifile = fopen (infilename, "rb")) != NULL)
  57. {
  58. fseek (ifile, 0L, 2);
  59. file_size = (unsigned long) ftell (ifile);
  60. fseek (ifile, 0L, 0);
  61. get_frequency_count ();
  62. build_initial_heap ();
  63. build_code_tree ();
  64. if (!generate_code_table ())
  65. {
  66. printf ("ERROR!  Code Value Out of Range. Cannot Compress.n");
  67. return 0;
  68. }
  69. else
  70. {
  71. if ((ofile = fopen (outfilename, "wb")) != NULL)
  72. {
  73. fwrite (&file_size, sizeof (file_size), 1, ofile);
  74. fwrite (code, 2, 256, ofile);
  75. fwrite (code_length, 1, 256, ofile);
  76. fseek (ifile, 0L, 0);
  77. compress_image ();
  78. fclose (ofile);
  79. }
  80. else
  81. {
  82. printf("nERROR: Couldn't create output file %sn", outfilename);
  83. return 0;
  84. }
  85. }
  86. fclose (ifile);
  87. }
  88. else
  89. {
  90. printf ("nERROR:  %s -- File not found!n", infilename);
  91. return 0;
  92. }
  93. return 1;
  94. }
  95. /**************************************************************************
  96.  COMPRESS_IMAGE ()
  97.  本函数进行数据的压缩操作
  98.  **************************************************************************/
  99. void compress_image ()
  100. {
  101.    register unsigned int    thebyte = 0;
  102.    register short           loop1;
  103.    register unsigned short  current_code;
  104.    register unsigned long   loop;
  105.    unsigned short  current_length, dvalue;
  106.    unsigned long   curbyte = 0;
  107.    short           curbit = 7;
  108.    for (loop = 0L; loop < file_size; loop++)
  109.    {
  110.       dvalue         = (unsigned short) getc (ifile);
  111.       current_code   = code[dvalue];
  112.       current_length = (unsigned short) code_length[dvalue];
  113.       for (loop1 = current_length-1; loop1 >= 0; --loop1)
  114.       {
  115.          if ((current_code >> loop1) & 1)
  116.             thebyte |= (char) (1 << curbit);
  117.          if (--curbit < 0)
  118.          {
  119.             putc (thebyte, ofile);
  120.             thebyte = 0;
  121.             curbyte++;
  122.             curbit = 7;
  123.          }
  124.       }
  125.    }
  126.    putc (thebyte, ofile);
  127.    compress_charcount = ++curbyte;
  128. }
  129. /**************************************************************************
  130.  GENERATE_CODE_TABLE ()
  131.  本函数生成压缩码表
  132.  **************************************************************************/
  133. unsigned short  generate_code_table ()
  134. {
  135.    register unsigned short  loop;
  136.    register unsigned short  current_length;
  137.    register unsigned short  current_bit;
  138.    unsigned short  bitcode;
  139.    short           parent;
  140.    for (loop = 0; loop < 256; loop++)
  141.       if (frequency_count[loop])
  142.       {
  143.          current_length = bitcode = 0;
  144.          current_bit = 1;
  145.          parent = father[loop];
  146.          while (parent)
  147.          {
  148.             if (parent < 0)
  149.             {
  150.                bitcode += current_bit;
  151.                parent = -parent;
  152.             }
  153.             parent = father[parent];
  154.             current_bit <<= 1;
  155.             current_length++;
  156.          }
  157.          code[loop] = bitcode;
  158.          if (current_length > 16)
  159.             return (0);
  160.          else
  161.             code_length[loop] = (unsigned char) current_length;
  162.       }
  163.       else
  164.          code[loop] = code_length[loop] = 0;
  165.    return (1);
  166. }
  167. /**************************************************************************
  168.  BUILD_CODE_TREE ()
  169.  建立压缩编码树
  170.  **************************************************************************/
  171. void build_code_tree ()
  172. {
  173.    void    reheap ();
  174.    register unsigned short  findex;
  175.    register unsigned long   heap_value;
  176.    while (heap_length != 1)
  177.    {
  178.       heap_value = heap[1];
  179.       heap[1]    = heap[heap_length--];
  180.       reheap (1);
  181.       findex = heap_length + 255;
  182.       frequency_count[findex] = frequency_count[heap[1]] +
  183.                                 frequency_count[heap_value];
  184.       father[heap_value] =  findex;
  185.       father[heap[1]]    = -findex;
  186.       heap[1]            =  findex;
  187.       reheap (1);
  188.    }
  189.    father[256] = 0;
  190. }
  191. /**************************************************************************
  192.  REHEAP ()
  193.  从当前的堆树结构中建立逻辑堆结构
  194.  **************************************************************************/
  195. void reheap (heap_entry)
  196. unsigned short  heap_entry;
  197. {
  198.    register unsigned short  index;
  199.    register unsigned short  flag = 1;
  200.    unsigned long   heap_value;
  201.    heap_value = heap[heap_entry];
  202.    while ((heap_entry <= (heap_length >> 1)) && (flag))
  203.    {
  204.       index = heap_entry << 1;
  205.       if (index < heap_length)
  206.          if (frequency_count[heap[index]] >= frequency_count[heap[index+1]])
  207.             index++;
  208.       if (frequency_count[heap_value] < frequency_count[heap[index]])
  209.  flag--;
  210.       else
  211.       {
  212.          heap[heap_entry] = heap[index];
  213.          heap_entry       = index;
  214.       }
  215.    }
  216.    heap[heap_entry] = heap_value;
  217. }
  218. /**************************************************************************
  219.  BUILD_INITIAL_HEAP ()
  220.  本函数从初始的频率统计数据中建立堆结构
  221.  **************************************************************************/
  222. void build_initial_heap ()
  223. {
  224.    void    reheap ();
  225.    register unsigned short  loop;
  226.    heap_length = 0;
  227.    for (loop = 0; loop < 256; loop++)
  228.       if (frequency_count[loop])
  229.          heap[++heap_length] = (unsigned long) loop;
  230.    for (loop = heap_length; loop > 0; loop--)
  231.       reheap (loop);
  232. }
  233. /**************************************************************************
  234.  GET_FREQUENCY_COUNT ()
  235.  本函数对需要进行压缩的数据进行频率统计
  236.  **************************************************************************/
  237. void get_frequency_count ()
  238. {
  239.    register unsigned long  loop;
  240.    for (loop = 0; loop < file_size; loop++)
  241.       frequency_count[getc (ifile)]++;
  242. }
  243. /**************************************************************************
  244.  Huffman_Decompression ()
  245.  本函数进行Huffman解码
  246.  **************************************************************************/
  247. int  Huffman_Decompression(char * infilename, char * outfilename)
  248. {
  249. if ((ifile = fopen (infilename, "rb")) != NULL)
  250. {
  251. fread (&file_size, sizeof (file_size), 1, ifile);
  252. fread (code, 2, 256, ifile);
  253. fread (code_length, 1, 256, ifile);
  254. build_decomp_tree ();
  255. if ((ofile = fopen (outfilename, "wb")) != NULL)
  256. {
  257. decompress_image();
  258. fclose (ofile);
  259. }
  260. else
  261. {
  262. printf ("nERROR:  Couldn't create output file %sn", outfilename);
  263. return 0;
  264. }
  265. fclose (ifile);
  266. }
  267. else
  268. {
  269. printf ("nERROR:  %s -- File not found!n", infilename);
  270. return 0;
  271. }
  272. return 1;
  273. }
  274. /**************************************************************************
  275.  BUILD_DECOMP_TREE ()
  276.  创建解压缩树
  277.  **************************************************************************/
  278. void  build_decomp_tree ()
  279. {
  280.    register unsigned short  loop1;
  281.    register unsigned short  current_index;
  282.    unsigned short  loop;
  283.    unsigned short  current_node = 1;
  284.    decomp_tree[1] = 1;
  285.    for (loop = 0; loop < 256; loop++)
  286.    {
  287.       if (code_length[loop])
  288.       {
  289.  current_index = 1;
  290.  for (loop1 = code_length[loop] - 1; loop1 > 0; loop1--)
  291.  {
  292.     current_index = (decomp_tree[current_index] << 1) +
  293.     ((code[loop] >> loop1) & 1);
  294.     if (!(decomp_tree[current_index]))
  295.        decomp_tree[current_index] = ++current_node;
  296.  }
  297.  decomp_tree[(decomp_tree[current_index] << 1) +
  298.    (code[loop] & 1)] = -loop;
  299.       }
  300.    }
  301. }
  302. /**************************************************************************
  303.  DECOMPRESS_IMAGE ()
  304.  进行解压缩操作
  305.  **************************************************************************/
  306. void  decompress_image ()
  307. {
  308.    register unsigned short  cindex = 1;
  309.    register char            curchar;
  310.    register short           bitshift;
  311.    unsigned long  charcount = 0L;
  312.    while (charcount < file_size)
  313.    {
  314.       curchar = (char) getc (ifile);
  315.       for (bitshift = 7; bitshift >= 0; --bitshift)
  316.       {
  317.  cindex = (cindex << 1) + ((curchar >> bitshift) & 1);
  318.  if (decomp_tree[cindex] <= 0)
  319.  {
  320.     putc ((int) (-decomp_tree[cindex]), ofile);
  321.     if ((++charcount) == file_size)
  322.                bitshift = 0;
  323.             else
  324.                cindex = 1;
  325.  }
  326.  else
  327.     cindex = decomp_tree[cindex];
  328.       }
  329.    }
  330. }