#include <stdio.h>
#include <stdlib.h>
#include <math.h>
#include<string.h>
typedef struct
{
char CH;
int weight;
int parent, lchild, rchild;
}DATA;
typedef struct
{
char ch;
char code[30];
int cnt;
}codetype;
void Createtree(DATA *hfmTree, int N)
{
int i, j, min, cmin;
int m, c;
hfmTree[0].CH = ' ';
hfmTree[0].parent = hfmTree[0].lchild = hfmTree[0].rchild = -1;
scanf("%d", &hfmTree[0].weight);
for (i = 1; i<N; i++)
{
hfmTree[i].CH = 'A' + i - 1;
hfmTree[i].parent = hfmTree[i].lchild = hfmTree[i].rchild = -1;
scanf("%d", &hfmTree[i].weight);
}
for (i = N; i<2 * N - 1; i++)
{
min = 99999;
cmin = 99999;
m = 0; c = 0;
for (j = 0; j<i; j++)
{
if (hfmTree[j].parent == -1)
if (hfmTree[j].weight<min)
{
c = m;
cmin = min;
min = hfmTree[j].weight;
m = j;
}
else if (hfmTree[j].weight<cmin)
{
cmin = hfmTree[j].weight;
c = j;
}
}
hfmTree[i].weight = min + cmin;
hfmTree[i].CH = ' ';
hfmTree[i].lchild = m;
hfmTree[i].rchild = c;
hfmTree[m].parent = i;
hfmTree[c].parent = i;
hfmTree[i].parent = -1;
}
}
void Hfmcode(DATA *hfmTree, codetype *codeFile, int N)
{
int i, p, c;
codetype S;
for (i = 0; i<N; i++)
{
c = i;
p = hfmTree[c].parent;
S.ch = hfmTree[i].CH;
S.cnt = N;
S.code[N] = '\0';
while (p != -1)
{
if (hfmTree[p].lchild == c)
S.code[--S.cnt] = '0';
else
S.code[--S.cnt] = '1';
c = p;
p = hfmTree[c].parent;
}
codeFile[i] = S;
}
}
void Decode(DATA *hfmTree, char *ToBeTran, int N)
{
int i, j = 0, ct = 0;
FILE *fp;
fp = fopen("codeFile.txt", "r");
char str[200], ch;
fscanf(fp, "%s", str);
ch = str[0];
i = 2 * N - 2;
while (1)
{
if (ch == '0')
i = hfmTree[i].lchild;
else if (ch == '1')
i = hfmTree[i].rchild;
if (hfmTree[i].lchild == -1 || hfmTree[i].rchild == -1)
{
ToBeTran[ct++] = hfmTree[i].CH;
i = 2 * N - 2;
}
j++;
ch = str[j];
if (j == strlen(str))
break;
}
if ((hfmTree[i].lchild != -1 || hfmTree[i].rchild != -1) && i != 2 * N - 2)
printf("编码有误!");
ToBeTran[ct] = '\0';
fclose(fp);
}
int main()
{
int N;
int i, j;
FILE *fp, *fp1, *fp2;
char str[200];
char *ToBeTran, c;
DATA *hfmTree;
codetype *codeFile;
printf("字符集大小:");
scanf("%d", &N);
ToBeTran = (char *)malloc(sizeof(char) * 40);
codeFile = (codetype *)malloc(sizeof(codetype)*N);
hfmTree = (DATA *)malloc(sizeof(DATA)*(2 * N - 1));
fp = fopen("fmTree.txt", "w");
if (fp == NULL)
return 0;
fp1 = fopen("codeFile.txt", "w");
if (fp1 == NULL)
return 0;
fp2 = fopen("TextFile.txt", "w");
if (fp2 == NULL)
return 0;
printf("输入空格和字母的频度:\n");
Createtree(hfmTree, N);
Hfmcode(hfmTree, codeFile, N);
scanf("%c", &c);
printf("请输入需要编码的字符串:\n");
gets(str);
printf("\n");
fprintf(fp, "%-8s%-8s%-8s%-8s%-8s%-8s%-8s\n", "单元号", "字符", "权值", "双亲", "左孩子", "右孩子");
for (i = 0; i<2 * N - 1; i++)
fprintf(fp, "%-8d%-8c%-8d%-8d%-8d%-8d\n", i, hfmTree[i].CH, hfmTree[i].weight, hfmTree[i].parent, hfmTree[i].lchild, hfmTree[i].rchild);
fclose(fp);
printf("哈夫曼树信息表以存入fmTree.txt文档中..\n");
for (i = 0; i<strlen(str); i++)
{
if (str[i] == ' ')
fprintf(fp1, "%s", codeFile[0].code + codeFile[0].cnt);
else
fprintf(fp1, "%s", codeFile[str[i] - 'A' + 1].code + codeFile[str[i] - 'A' + 1].cnt);
}
fclose(fp1);
printf("该字符串编码以存入codeFile.txt文档中..:\n");
Decode(hfmTree, ToBeTran, N);
fprintf(fp2, "%s", ToBeTran);
fclose(fp2);
printf("编码译文以存入TextFile.txt文档中..:\n");
return 0;
}