Преобразование MTF — различия между версиями
Строка 5: | Строка 5: | ||
== Описание алгоритма == | == Описание алгоритма == | ||
− | Изначально каждое возможное значение байта записывается в список (алфавит), в ячейку с номером, равным значению байта, т.е. (0, 1, 2, 3, | + | Изначально каждое возможное значение байта записывается в список (алфавит), в ячейку с номером, равным значению байта, т.е. <tex>(0, 1, 2, 3, \dots, 255)</tex>. В процессе обработки данных этот список изменяется. По мере поступления очередного символа на выход подается номер элемента, содержащего его значение. После чего этот символ перемещается в начало списка, смещая остальные элементы вправо. |
− | Современные алгоритмы (например, bzip2<ref>[http://ru.wikipedia.org/wiki/Bzip2 {{---}} bzip2]</ref>) перед алгоритмом MTF используют [[преобразование Барроуза-Уиллера|алгоритм BWT]], поэтому в качестве примера рассмотрим строку <tex> | + | Современные алгоритмы (например, bzip2<ref>[http://ru.wikipedia.org/wiki/Bzip2 {{---}} bzip2]</ref>) перед алгоритмом MTF используют [[преобразование Барроуза-Уиллера|алгоритм BWT]], поэтому в качестве примера рассмотрим строку <tex>S = </tex>''"BCABAAA"'', полученную из строки ''"ABACABA"'' в результате [[Преобразование Барроуза-Уиллера|преобразования Барроуза-Уиллера]]. Первый символ строки <tex>S</tex> 'B' является вторым элементом алфавита ''"ABC"'', поэтому на вывод подаётся <tex>1</tex>. После перемещения 'B' в начало алфавита тот принимает вид ''"BAC"''. Дальнейшая работа алгоритма показана в таблице: |
{| class="wikitable" | {| class="wikitable" | ||
Строка 27: | Строка 27: | ||
|} | |} | ||
− | Таким образом, результат работы алгоритма: <tex>MTF( | + | Таким образом, результат работы алгоритма: <tex>MTF(S) = </tex> ''"1222100"''. |
− | Вот примерная реализация этого алгоритма. Здесь массив <tex>\mathtt{alphabet}</tex> хранит количество символов перед символом <tex> | + | Вот примерная реализация этого алгоритма. Здесь массив <tex>\mathtt{alphabet}</tex> хранит количество символов перед символом <tex>S[i]</tex>, <tex>N</tex> {{---}} длина строки <tex>S</tex>. |
<code> | <code> | ||
Строка 40: | Строка 40: | ||
</code> | </code> | ||
− | Данный алгоритм работает за <tex>O( | + | Данный алгоритм работает за <tex>O(N \cdot M)</tex>, где <tex>N</tex> {{---}} размер алфавита, <tex>M</tex> {{---}} длина строки, что не очень быстро. Этот алгоритм можно реализовать за <tex>O(N\log(N+M))</tex>. |
== Описание алгоритма за O(N log(N+M)) == | == Описание алгоритма за O(N log(N+M)) == | ||
− | Пусть дан алфавит размером <tex> | + | Пусть дан алфавит размером <tex>M</tex> и строка <tex>S</tex> длиной <tex>N</tex>. Заведем массив <tex>\mathtt{used}[1..N+M]</tex> и последние <tex>M</tex> ячеек заполним единицами. Запомним для каждого символа алфавита позицию в нашем массиве. Например, <tex>\mathtt{alphabet}['a'] = N+1</tex>, <tex>\mathtt{alphabet}['b'] = N+2</tex>, ... , <tex>\mathtt{alphabet}['z'] = N+M</tex>. |
− | При обработке <tex> | + | При обработке <tex>i</tex>-го символа посчитаем и выпишем сумму на отрезке <tex>[1, \mathtt{alphabet}[S[i]] - 1]</tex>, поменяем значения ячеек <tex>\mathtt{used}[N-i+1]</tex> и <tex>\mathtt{used}[\mathtt{alphabet}[S[i]]]</tex> местами, также стоит поменять значение в ячейке <tex>\mathtt{alphabet}[S[i]]</tex> на <tex>N-i+1</tex>. |
<code> | <code> | ||
Строка 61: | Строка 61: | ||
</code> | </code> | ||
− | Функцию <tex>sum</tex> можно реализовывать по-разному. | + | Функцию <tex>\mathtt{sum}</tex> можно реализовывать по-разному. |
<code> | <code> | ||
Строка 71: | Строка 71: | ||
</code> | </code> | ||
− | Такая реализация работает за <tex>O(right-left)</tex>, общая сложность алгоритма равна <tex>O( | + | Такая реализация работает за <tex>O(right-left)</tex>, общая сложность алгоритма равна <tex>O(N \cdot M)</tex> Но можно находить сумму на отрезке при помощи [[Дерево_отрезков._Построение | дерева отрезков]], что сократит время работы до <tex>O(\log(right-left))</tex>. Итого, общая сложность будет равна <tex>O(N\log(N+M))</tex> |
== Обратное преобразование == | == Обратное преобразование == | ||
− | Пусть даны строка <tex> | + | Пусть даны строка <tex>S = </tex>''"1222100"'' и исходный алфавит ''"ABC"''. Символ с номером <tex>1</tex> в алфавите {{---}} это 'B'. На вывод подаётся 'B', и этот символ перемещается в начало алфавита. Символ с номером <tex>2</tex> в алфавите {{---}} это 'A', поэтому 'A' подается на вывод и перемещается в начало алфавита. Дальнейшее преобразование происходит аналогично. |
{| class ="wikitable" | {| class ="wikitable" | ||
Строка 95: | Строка 95: | ||
|} | |} | ||
− | Значит, исходная строка <tex>MTF^{-1}( | + | Значит, исходная строка <tex>MTF^{-1}(S) = </tex>''"BCABAAA"''. |
== Применение == | == Применение == | ||
Строка 127: | Строка 127: | ||
− | == | + | == Источники информации == |
− | + | # [http://compression.ru/arctest/descript/bwt-faq.htm Burrows Wheeler Transform FAQ] | |
− | + | # [http://ru.wikipedia.org/wiki/Move-To-Front Move-To-Front (Википедия)] | |
− | |||
− | |||
− | |||
# Ryabko, B. Ya. ''Data compression by means of a «book stack»'', Problems of Information Transmission, 1980, v. 16: (4), pp. 265–269. | # Ryabko, B. Ya. ''Data compression by means of a «book stack»'', Problems of Information Transmission, 1980, v. 16: (4), pp. 265–269. | ||
# Ryabko, B. Ya.; Horspool, R. Nigel; Cormack, Gordon V. Comments to: ''«A locally adaptive data compression scheme»'' by J. L. Bentley, D. D. Sleator, R. E. Tarjan and V. K. Wei. Comm. ACM 30 (1987), no. 9, 792—794. | # Ryabko, B. Ya.; Horspool, R. Nigel; Cormack, Gordon V. Comments to: ''«A locally adaptive data compression scheme»'' by J. L. Bentley, D. D. Sleator, R. E. Tarjan and V. K. Wei. Comm. ACM 30 (1987), no. 9, 792—794. |
Версия 19:59, 15 января 2015
Определение: |
Преобразование MTF (англ. move-to-front, движение к началу) — алгоритм кодирования, используемый для предварительной обработки данных (обычно потока байтов) перед сжатием, разработанный для улучшения эффективности последующего кодирования. |
Содержание
Описание алгоритма
Изначально каждое возможное значение байта записывается в список (алфавит), в ячейку с номером, равным значению байта, т.е.
. В процессе обработки данных этот список изменяется. По мере поступления очередного символа на выход подается номер элемента, содержащего его значение. После чего этот символ перемещается в начало списка, смещая остальные элементы вправо.Современные алгоритмы (например, bzip2[1]) перед алгоритмом MTF используют алгоритм BWT, поэтому в качестве примера рассмотрим строку "BCABAAA", полученную из строки "ABACABA" в результате преобразования Барроуза-Уиллера. Первый символ строки 'B' является вторым элементом алфавита "ABC", поэтому на вывод подаётся . После перемещения 'B' в начало алфавита тот принимает вид "BAC". Дальнейшая работа алгоритма показана в таблице:
Символ | Список | Вывод |
---|---|---|
B | ABC | 1 |
C | BAC | 2 |
A | CBA | 2 |
B | ACB | 2 |
A | BAC | 1 |
A | ABC | 0 |
A | ABC | 0 |
Таким образом, результат работы алгоритма:
"1222100".Вот примерная реализация этого алгоритма. Здесь массив
хранит количество символов перед символом , — длина строки .
list<int> mtf(N): list<int> result(N) for i = 1 to N result.append(alphabet[S[i]]) помещаем символ S[i] в начало алфавита return result
Данный алгоритм работает за
, где — размер алфавита, — длина строки, что не очень быстро. Этот алгоритм можно реализовать за .Описание алгоритма за O(N log(N+M))
Пусть дан алфавит размером
и строка длиной . Заведем массив и последние ячеек заполним единицами. Запомним для каждого символа алфавита позицию в нашем массиве. Например, , , ... , .При обработке
-го символа посчитаем и выпишем сумму на отрезке , поменяем значения ячеек и местами, также стоит поменять значение в ячейке на .
list<int> mtf(N): list<int> result(N) list<int> used(N+M) for i = 1 to M //Заполняем последние M ячеек единицами used[i+N] = 1 for i = 1 to N result.append(sum(1, alphabet[S[i]] - 1)) //Запоминаем ответ swap(used[N-i+1], used[alphabet[S[i]]]) //Меняем значения alphabet[S[i]] = N-i+1 //Изменяем позицию символа в массиве return result
Функцию
можно реализовывать по-разному.
int sum(left, right) result = 0 for i = left to right result = result + used[i] return result
Такая реализация работает за дерева отрезков, что сократит время работы до . Итого, общая сложность будет равна
, общая сложность алгоритма равна Но можно находить сумму на отрезке при помощиОбратное преобразование
Пусть даны строка
"1222100" и исходный алфавит "ABC". Символ с номером в алфавите — это 'B'. На вывод подаётся 'B', и этот символ перемещается в начало алфавита. Символ с номером в алфавите — это 'A', поэтому 'A' подается на вывод и перемещается в начало алфавита. Дальнейшее преобразование происходит аналогично.Символ | Список | Вывод |
---|---|---|
1 | ABC | B |
2 | BAC | C |
2 | CBA | A |
2 | ACB | B |
1 | BAC | A |
0 | ABC | A |
0 | ABC | A |
Значит, исходная строка
"BCABAAA".Применение
Этот метод позволяет легко преобразовать данные, насыщенные длинными повторами разных символов в блок данных, самыми частыми символами которого будут нули. Без MTF нас подстерегают разного рода трудности в решении проблемы адаптации к данным, поскольку в разных местах данных, полученных на выходе BWT-преобразования, разные символы являются преобладающими. Зачастую мы встречаемся с последовательностями типа "bbbbbcccccdddddaaaaa".
Попробуем сжать эту последовательность при помощи, например, метода Хаффмана. Вероятности всех четырех символов в данном примере равны . Легко посчитать, что в результате кодирования мы получим последовательность длиной бит.
Теперь проделаем то же самое со строкой, подвергнутой MTF-преобразованию (предположим, начальный алфавит выглядит как "abcd").
"bbbbbcccccdddddaaaaa" — исходная строка
"10000200003000030000" — строка после MTF
Символ | Частота | Вероятность | Код Хаффмана |
---|---|---|---|
0 | 16 | 4/5 | 0 |
1 | 2 | 1/10 | 10 |
2 | 1 | 1/20 | 110 |
3 | 1 | 1/20 | 111 |
В результате сжатия получаем последовательность длиной арифметического кодирования для данного примера будет еще значительней.
бит. Стоит заметить, что выигрыш от примененияПримечания
Источники информации
- Burrows Wheeler Transform FAQ
- Move-To-Front (Википедия)
- Ryabko, B. Ya. Data compression by means of a «book stack», Problems of Information Transmission, 1980, v. 16: (4), pp. 265–269.
- Ryabko, B. Ya.; Horspool, R. Nigel; Cormack, Gordon V. Comments to: «A locally adaptive data compression scheme» by J. L. Bentley, D. D. Sleator, R. E. Tarjan and V. K. Wei. Comm. ACM 30 (1987), no. 9, 792—794.