PackBits - Википедия - PackBits

PackBits это быстрый, простой сжатие без потерь схема для кодирование длин серий данных.

яблоко представил формат PackBits с выпуском MacPaint на Macintosh компьютер. Эта схема сжатия является одним из типов сжатия, которое можно использовать в TIFF -файлы. TGA -files также используют эту схему сжатия RLE, но обрабатывают поток данных как пиксели, а не байты.

Поток данных PackBits состоит из пакетов с однобайтовым заголовком, за которым следуют данные. Заголовок - это байт со знаком; данные могут быть подписаны, без подписи или упакованы (например, пиксели MacPaint).

В следующей таблице п - значение байта заголовка в виде целого числа со знаком.

Байт заголовкаДанные, следующие за байтом заголовка
От 0 до 127(1 + п) буквальный байты данных
От −1 до −127Один байт данных, повторяющийся (1 - п) раз в распакованном выходе
−128Нет операции (пропускать и рассматривать следующий байт как байт заголовка)

Обратите внимание, что интерпретация 0 как положительного или отрицательного значения не влияет на результат. Выполнения двух байтов, смежных с невыполнениями, обычно записываются как буквальные данные. На основе данных PackBits невозможно определить конец потока данных; то есть, нужно знать размер сжатых или несжатых данных до чтения потока данных PackBits, чтобы знать, где он заканчивается.

Apple Computer (см. Внешнюю ссылку) предоставляет этот короткий пример упакованных данных:FE AA 02 80 00 2A FD AA 03 80 00 2A 22 F7 AA

Следующий код, написанный на Microsoft VBA, распаковывает данные:

Sub РаспаковатьBitsDemo()   Тусклый Файл В качестве Вариант   Тусклый MyOutput В качестве Нить   Тусклый Считать В качестве Длинный   Тусклый я В качестве Длинный, j В качестве Длинный     Файл = «FE AA 02 80 00 2A FD AA 03 80 00 2A 22 F7 AA»   Файл = Расколоть(Файл, " ")      За я = LBound(Файл) К UBound(Файл)      Считать = Заявление.Рабочий лист.Hex2Dec(Файл(я))      Выбирать Дело Считать      Дело Является >= 128         Считать = 256 - Считать Дополнение до двух         За j = 0 К Считать 'с нуля            MyOutput = MyOutput & Файл(я + 1) & " "         Следующий j         я = я + 1 'Отрегулируйте указатель      Дело Еще         За j = 0 К Считать 'с нуля            MyOutput = MyOutput & Файл(я + j + 1) & " "         Следующий j         я = я + j 'Отрегулируйте указатель      Конец Выбирать   Следующий я   Отлаживать.Распечатать MyOutput   'AA AA AA 80 00 2A AA AA AA AA 80 00 2A 22 AA AA AA AA AA AA AA AA AA AAКонец Sub

Та же реализация в JS:

/** * Вспомогательные функции для создания удобочитаемого ввода и вывода *  * Также см. Эту скрипку для интерактивного декодера PackBits: * https://jsfiddle.net/volter9/tj04ejdt/ */функция str2hex (ул) {    возвращаться ул.расколоть('').карта(функция (char) {        вар ценить = char.charCodeAt(0);        возвращаться ((ценить < 16 ? '0' : '') + ценить.нанизывать(16)).toUpperCase();    }).присоединиться(' ');}функция hex2str (шестнадцатеричный) {    возвращаться шестнадцатеричный.расколоть(' ').карта(функция (нить) {        возвращаться Нить.fromCharCode(parseInt(нить, 16));    }).присоединиться('');}/** * Функция распаковки PackBits *  * @param {String} данные * @return {String} */функция unpackBits (данные) {    вар выход = '',    я = 0;    пока (я < данные.длина) {        вар шестнадцатеричный = данные.charCodeAt(я);        если (шестнадцатеричный >= 128) {            шестнадцатеричный = 256 - шестнадцатеричный;            за (вар j = 0; j <= шестнадцатеричный; ++j) {                выход += данные.диаграмма(я + 1);            }            ++я;        }        еще {            за (вар j = 0; j <= шестнадцатеричный; ++j) {                выход += данные.диаграмма(я + j + 1);            }            я += j;        }        ++я;    }    возвращаться выход;}вар оригинал = 'FE AA 02 80 00 2A FD AA 03 80 00 2A 22 F7 AA',    данные = unpackBits(hex2str(оригинал));// Вывод: AA AA AA 80 00 2A AA AA AA AA 80 00 2A 22 AA AA AA AA AA AA AA AA AA AAконсоль.бревно(str2hex(данные));

внешняя ссылка