Compression-Project/src/huffman.cpp
2020-05-30 19:29:38 +00:00

80 lines
2.2 KiB
C++

#include "huffman.h"
#include "common.h"
namespace huffman {
struct node {
char c; int f; node * l, * r;
node(char _c, int _f) { c = _c, f = _f, l = r = NULL; }
};
struct comp { bool operator()(node * a, node * b) { return a->f > b->f; } };
node * root;
vector<bool> path, code[128];
void traverse(node * n) {
if (n->l) path.push_back(0), traverse(n->l), path.pop_back();
if (n->r) path.push_back(1), traverse(n->r), path.pop_back();
if (n->c) code[n->c] = path;
}
void encode_tree(node * n, vector<bool> & v) {
if (n->c) {
v.push_back(1);
for (int i = 0; i < 7; ++i) v.push_back(1 & (n->c >> i));
}
else {
v.push_back(0);
n->l ? (v.push_back(1), encode_tree(n->l, v)) : v.push_back(0);
n->r ? (v.push_back(1), encode_tree(n->r, v)) : v.push_back(0);
}
}
int idx = 0;
void decode_tree(node * n, vector<bool> & v) {
if (idx > v.size()) return;
if (v[idx++] == 1) {
for (int i = 0; i < 7; ++i) if (v[idx++]) n->c |= (1 << i);
}
else {
if (v[idx++] == 1) n->l = new node(0, 0), decode_tree(n->l, v);
if (v[idx++] == 1) n->r = new node(0, 0), decode_tree(n->r, v);
}
}
void generate(vector<int> f) {
priority_queue<node *, vector<node *>, comp> pq;
for (int c = 1; c < 128; ++c) if (f[c]) {
node * n = new node(c, f[c]); pq.push(n);
}
while (pq.size() > 1) {
node * l = pq.top(); pq.pop(); node * r = pq.top(); pq.pop();
node * n = new node(0, l->f + r->f); n->l = l, n->r = r;
pq.push(n);
}
root = pq.top();
}
void solve(node * n, vector<bool> & v, string & s) {
if (idx > v.size() || !n) return;
if (n->c) s += n->c, solve(root, v, s);
else v[idx++] == 0 ? solve(n->l, v, s) : solve(n->r, v, s);
}
vector<bool> encode(string s) {
vector<int> f(128, 0);
for (auto& c : s) ++f[c];
generate(f);
traverse(root);
vector<bool> ret;
encode_tree(root, ret);
for (auto& c : s) for (auto b : code[c]) ret.push_back(b);
return ret;
}
string decode(vector<bool> v) {
root = new node(0, 0);
decode_tree(root, v);
string ret;
while (idx < v.size()) solve(root, v, ret);
return ret;
}
}