ハフマン符号
ハフマン符号(ハフマンふごう、英: Huffman coding)とは、1952年にデビッド・ハフマンによって開発された符号で、文字列をはじめとするデータの可逆圧縮などに使用される[1][2]。
ほかのエントロピー符号と同様、よく出現する文字には短いビット列を、あまり出現しない文字には長いビット列を割り当てることで、メッセージ全体の符号化に使われるデータ量を削減することを狙っている。
コンパクト符号やエントロピー符号の一つ。JPEGやZIP (Deflate) などの圧縮フォーマットで使用されている。
シャノン符号化が最適ではない場合が存在する不完全な符号であったのに対し、ハフマン符号は(整数の符号語長という制約のもとでは、)常に最適な符号を構成できる。擬似的に実数の符号語長を割り振る算術符号と比較すれば、データ圧縮効率は劣る。ただし、算術符号やその他の高効率の符号化法と異なり、特許の問題が無い。
種類
[編集]符号化の原理上、木を構成する前に各記号の出現頻度をあらかじめ知っておく必要がある。
ファイルなど固定長のデータに対し、1度全部読み込んで、データのすべての記号を調べて符号木を構築しておき、もう1度頭から読み直して符号化を行う方法が、静的ハフマン符号 (Static Huffman coding) である。
deflateでは、データの全部ではなく、ブロック毎に符号を作っている。これをdeflateではダイナミックハフマン符号(英: Dynamic Huffman coding)と呼んでいる。
一方、最初の状態では頻度情報を持たず、記号を1個読み込むごとに符号木を作り直す方法は適応型ハフマン符号 (Adaptive Huffman coding) である。これを指して日本では動的ハフマン符号と呼ぶことが多い。
符号化の原理
[編集]例として DAEBCBACBBBC という12文字からなるメッセージを考える。
このメッセージ中には5種類の文字が使われているので、もしそれぞれの文字を固定長のビット列で表すとすれば、1文字につき最低3ビット必要となる。
文字とビット列の対応表として、
A B C D E 000 001 010 011 100
を使い、それぞれの文字をビット列に置き換えたとすると、メッセージ全体は次のビット列で表される。
D A E B C B A C B B B C 011 000 100 001 010 001 000 010 001 001 001 010
12文字 x 3ビットで全体のビット数は36ビットとなる。
もし、よく出現する文字には短いビット列を、あまり出現しない文字には長いビット列を割り当てることができれば、メッセージ全体の符号化に必要なビット数を抑えることが期待できる。そこで、文字とビット列の対応表として、
A B C D E 110 0 10 1110 1111
を使ったとすると、メッセージ全体は
D A E B C B A C B B B C 1110 110 1111 0 10 0 110 10 0 0 0 10
となり、全体のビット数は25ビットとなる。 固定長の方式に比べると 70% ほどのデータ量に抑えられている。
ハフマン符号の構成
[編集]文字とビット列の対応表を作るには、まずデータに出現する文字の出現回数を求め、それをもとにハフマン木と呼ばれるバイナリツリーを構成するという手順を踏む。
ハフマン木の構成の仕方は、次のようなアルゴリズムになる。
- まず、葉を含むすべての節点のうち、親を持たないものを集める。
- その中から、最小の値を持つものと2番目に小さい値を持つものを取り出す。
- それらを子供に持つ新しい節点を作る。このとき、新しい節点の値は、両方の子供の値の和とする。
以上を根節点に到達するまで繰り返すと、木が完成する。次に、根から順に左右に0と1の値を割り振っていく(左右のどちらに0と1を与えるかは任意)。すると、それぞれの葉(記号)に対して、一意にビット列が与えられる。この記号とビット列の関係をもとに、もとのデータの記号をビット列に変換していくことで符号化が行われる。
例
[編集]入力 DAEBCBACBBBC に対して上記のアルゴリズムを適用すると、
出現頻度と割り当てられた符号
文字 | 個数 | 符号 |
---|---|---|
B | 5 | 0 |
C | 3 | 10 |
A | 2 | 110 |
D | 1 | 1110 |
E | 1 | 1111 |
が得られ、個数の多い文字ほど短い符号が割り当てられていることがわかる。
接頭符号
[編集]ハフマン符号は、接頭符号である。つまり、ある文字に対応するビット列が、他の文字に対応するビット列の接頭辞になることはない。この特徴のおかげで、デコーダはビット列を先頭から一度読むだけで元のメッセージを一意にデコードできる。
利用例
[編集]算術符号やその他の高効率の符号化法と異なり、特許の問題が無い。 そのため、JPEGやZIPなどの圧縮フォーマットで使用されている。
実装
[編集]Groovyで実装を説明する。まず、符号情報とハフマン木のクラスを作る。
class CodeInfo { int code, codeSize }
class TreeNode implements Comparable<TreeNode> {
byte value; int occurrence; List<TreeNode> children;
int compareTo(TreeNode that) { this.occurrence - that.occurrence }
}
すると、バイト値→符号情報のテーブルは以下のようにして作れる。
CodeInfo[] createValue2Code(byte[] data) {
// 各バイト値の発生回数を数える
TreeNode[] nodes = new TreeNode[256]
for (int i in 0..<256) { nodes[i] = new TreeNode(value: i) }
for (byte v in data) { nodes[v].occurrence++ }
// ハフマン木を作成
PriorityQueue<TreeNode> queue = new PriorityQueue<TreeNode>(nodes as List)
while (queue.size() > 1) {
TreeNode n1 = queue.poll(), n2 = queue.poll()
queue << new TreeNode(occurrence: n1.occurrence + n2.occurrence, children: [n1, n2])
}
TreeNode root = queue.poll()
// 深さ優先探索でバイト値→符号情報を作成
CodeInfo[] value2code = new CodeInfo[256]
dfs(value2code, root, 0, 0)
return value2code
}
void dfs(CodeInfo[] value2code, TreeNode node, int code, int codeSize) {
if (node.children == null) {
value2code[node.value] = new CodeInfo(code: code, codeSize: codeSize)
} else {
dfs(value2code, node.children[0], code << 1, codeSize + 1)
dfs(value2code, node.children[1], (code << 1) + 1, codeSize + 1)
}
}
圧縮のために以下のようなビットストリームのクラスを作る。
class BitStream {
BitSet bs = new BitSet(); int len = 0, pos = 0;
void write(int v, int bits) {
for (int i in 0..<bits) { bs[len++] = ((v >>> i) & 1) != 0 }
}
int read(int bits) {
int v = 0; for (int i in 0..<bits) { if (bs[pos++]) { v |= 1 << i } }
return v
}
String toString() { "length = $len, {" + (0..<len).findAll({ bs[it] }).join(", ") + "}" }
}
すると、バイト値→符号情報のテーブルを使い以下のようにして圧縮できる。
void compress(BitStream bs, byte[] data, CodeInfo[] value2code) {
for (byte v in data) {
CodeInfo codeInfo = value2code[v]
bs.write(codeInfo.code, codeInfo.codeSize)
}
}
脚注
[編集]- ^ David A. Huffman, "A method for the construction of minimum-redundancy codes" (PDF), Proceedings of the I.R.E., Sept. 1952, pp. 1098-1102 (ハフマンのオリジナル論文)
- ^ 『誰が音楽をタダにした?』早川書房、2016年、22頁。