-
Notifications
You must be signed in to change notification settings - Fork 0
/
哈夫曼树.cpp
118 lines (118 loc) · 4.73 KB
/
哈夫曼树.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
//#define _CRT_SECURE_NO_WARNINGS 1
//#include <stdio.h>
//#include <malloc.h>
//#include <string.h>
//#define MAXVALUE 32767 //极大值相当于无穷大
//#define NODENUM 7 //叶子结点数
//typedef struct
//{
// char data; //数据域
// int weight; //结点的权值
// int parent, lch, rch; //双亲与孩子的下标
//}htNode, * huffmanTree;
//
//typedef char** huffmanCode; //第一个*是代表它是指针变量,说明它是数组
////第二个*说明它是指针数组,代表这个char类型数组里每个元素都是*huffmanCode变量
//
//int initHuffmanTree(huffmanTree& HT); //初始化哈夫曼树
//void creatHuffmanTree(huffmanTree& HT, int n); //构建哈夫曼树
//void createHuffmanCode(huffmanTree HT, huffmanCode& HC, int n); //编写哈夫曼编码
//int main()
//{
// huffmanTree HT;
// initHuffmanTree(HT);
// huffmanCode HC;
// creatHuffmanTree(HT, NODENUM);
// createHuffmanCode(HT, HC, NODENUM);
// /*for (int i = NODENUM + 1; i <= 2 * NODENUM - 1; i++)
// printf("%d ", HT[i].weight);*/
// for (int i = 1; i <= NODENUM; i++) //遍历输出编码
// {
// printf("%c:\t", HT[i].data);
// printf("%s\n", HC[i]);
// }
// return 0;
//}
//int initHuffmanTree(huffmanTree& HT)
//{
// HT = (htNode*)malloc(sizeof(htNode) * (2 * NODENUM)); //给HT分配2 * NODENUM个htNOde大小的htNode类型的数组
// for (int i = 1; i <= 2 * NODENUM - 1; i++) //下标从1开始到2 * NODENUM
// {
// HT[i].parent = HT[i].lch = HT[i].rch = -1; //双亲和孩子的值都置为-1
// }
// printf("请输入各个字母的频率数字!(空格间隔)\n");
// for (int i = 1; i <= NODENUM; i++) //权值只有1-n个
// {
// scanf("%d", &HT[i].weight); //给每个结点赋予权值
// }
// char c = getchar(); //这个来接收上面的回车
// printf("请输入字母名称!(连续输入,不用空格)\n");
// for (int i = 1; i <= NODENUM; i++)
// {
// //scanf("%c ",&HT[i].data);
// char a = getchar();
// if (a == '\n') //遇到回车就结束
// break;
// else
// HT[i].data = a; //给每个结点赋予数据
// }
//
// return 1;
//}
//
//void creatHuffmanTree(huffmanTree& HT, int n)
//{
// if (n <= 1) //如果结点数小于等于1,不创建
// return;
// int min1, min2; //定义两个数,来存储每次选取最小两个结点的权值
// int rnode, lnode; //定义两个下标值,来存储每次选取最小两个结点的下标
// for (int i = n + 1; i <= 2 * n - 1; i++) //要生成n-1个结点,所以要操作n—1次且从下标为n+1开始存储
// {
// int min1 = MAXVALUE; int lnode = -1; //让最小值初始化为极大值,这样叶子结点的最大值再大也不会超过这个值了
// int min2 = MAXVALUE; int rnode = -1;
// for (int j = 1; j <= i - 1; j++) //因为起先是在前n个中选择最小的两个结点的权值,但新生成一个后就得在前n+1个中选择最小的两个结点的权值
// { //假设n = 10 总结点数就得为19,那我们就只要比较18次就可以得出结果了,记住比较的次数比生成的总结点数少1
// if (HT[j].weight < min1 && HT[j].parent == -1) //这个小于就使得当出现相同的权值时优先考虑先出现的值,可以假设下
// {
// min2 = min1; rnode = lnode; //碰到比min1小的,那min1的值就给第二小的min2,下标也给它
// min1 = HT[j].weight; lnode = j; //然后最小的给min1,下标同理
// }
// else if (HT[j].weight < min2 && HT[j].parent == -1) //这是第二小的判断
// {
// min2 = HT[j].weight;
// rnode = j;
// }
// }
// HT[lnode].parent = HT[rnode].parent = i; //最小两个结点的parent变为生成结点的下标
// HT[i].lch = lnode; HT[i].rch = rnode; //生成结点的左孩子为最小的min1的下标,右孩子同理
// HT[i].weight = HT[lnode].weight + HT[rnode].weight; //生成结点的权值等于最小结点的权值相加
// }
//
//}
//
//void createHuffmanCode(huffmanTree HT, huffmanCode& HC, int n)
//{
// HC = (huffmanCode)malloc(sizeof(huffmanCode) * n + 1); //申请n + 1个huffmanCode大小huffmanCode类型的临时空间
// //因为下标是从一开始,所以我们要申请比结点多一个的结点,和哈夫曼树的结构对应,方便输出
// char* cd = (char*)malloc(sizeof(char) * n); //申请n个char大小char类型的临时空间,这个临时数组记录每次遍历出来的编码
// int start = 0, c = 0, f = 0; //start为cd数组记录下标,c初始为叶子结点下标,而后就是孩子结点的下标,f记录双亲结点的下标
// cd[n - 1] = '\0'; //这个就是给printf留着的,因为printf不会生成'\0',如果用puts就不用这句语句了
// for (int i = 1; i <= n; i++) //只要叶子结点的编码
// {
// start = n - 1; //这句要赋值n的话,start--要写在判断后方
// c = i;
// f = HT[c].parent;
// while (f != -1) //根节点没有双亲
// {
// start--;
// if (HT[f].lch == c) //是左孩子就是0,右孩子就为1
// cd[start] = '0';
// else
// cd[start] = '1';
// c = f; f = HT[c].parent; //向根结点接近
// }
// HC[i] = (char*)malloc(sizeof(char) * (n - start)); //给数组里的数组申请n - start个char大小的char*类型的临时空间
// strcpy(HC[i], &cd[start]); //cd里记录的编码给HC的第i个数组
// }
// free(cd); //释放临时空间
//}