forked from mapsme/omim
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhuffman.cpp
70 lines (61 loc) · 1.39 KB
/
huffman.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
#include "coding/huffman.hpp"
#include "base/logging.hpp"
namespace coding
{
HuffmanCoder::~HuffmanCoder()
{
DeleteHuffmanTree(m_root);
}
void HuffmanCoder::Init(vector<strings::UniString> const & data)
{
DeleteHuffmanTree(m_root);
BuildHuffmanTree(data.begin(), data.end());
BuildTables(m_root, 0);
}
bool HuffmanCoder::Encode(uint32_t symbol, Code & code) const
{
auto it = m_encoderTable.find(symbol);
if (it == m_encoderTable.end())
return false;
code = it->second;
return true;
}
bool HuffmanCoder::Decode(Code const & code, uint32_t & symbol) const
{
auto it = m_decoderTable.find(code);
if (it == m_decoderTable.end())
return false;
symbol = it->second;
return true;
}
void HuffmanCoder::BuildTables(Node * root, uint32_t path)
{
if (!root)
return;
if (root->isLeaf)
{
Code code(path, root->depth);
m_encoderTable[root->symbol] = code;
m_decoderTable[code] = root->symbol;
return;
}
BuildTables(root->l, path);
BuildTables(root->r, path + (static_cast<uint32_t>(1) << root->depth));
}
void HuffmanCoder::DeleteHuffmanTree(Node * root)
{
if (!root)
return;
DeleteHuffmanTree(root->l);
DeleteHuffmanTree(root->r);
delete root;
}
void HuffmanCoder::SetDepths(Node * root, uint32_t depth)
{
if (!root)
return;
root->depth = depth;
SetDepths(root->l, depth + 1);
SetDepths(root->r, depth + 1);
}
} // namespace coding