zoukankan      html  css  js  c++  java
  • [Andrew Stankevich's Contest#21] Lempel-Ziv Compression

    Time Limit: 20000/10000MS (Java/Others) Memory Limit: 128000/64000KB (Java/Others)     Special Judge

    Problem Description

          Most modern archivers, such as WinRAR or WinZIP, use modifications of Lempel-Ziv method as their primary compression algorithm. Although decompression of LZ-compressed archives is usually easy and fast, the compression process itself is often rather complicated and slow. Therefore professional archivers use approximation methods that sometimes do not allow to achieve the best possible compression. 

          This situation doesn’t satisfy your chief George. He would like to create the best archiver WinGOR. The archiver will use the following modification of LZ77 algorithm.
          The text is partitioned to chunks of length not exceeding 4096. Each chunk is compressed independently. We will describe the decompression of one chunk t. Based on this description, you will have to create a compression algorithm that will create the shortest possible compressed chunk x from the given chunk t.
          The compressed chunk is written down as the sequence of plain characters and repetition blocks. Plain character is 8 bits long. When decompressing, plain character c is simply copied to output. Repetition block (r; l) consists of two parts: reference r and length l, each 12 bits long. Reference r is an integer number between 1 and 4095. When repetition block (r; l) is obtained after decompressing i − 1 characters of text, characters t[i − r ... i − r + l − 1] are copied to output. Note, that r can be less than l, in this
    case recently copied characters are copied to output as well.

          To help decompressor distinguish between plain characters and repetition blocks a leading bit is prepended to each element of the compressed text: 0 means plain character follows, 1 — repetition block follows. 

          For example, “aaabbaaabababababab” can be compressed as “aaabb(5,4)(2,10)”. The compressed variant has 8 + 8 + 8 + 8 + 8 + 24 + 24 + 7 = 95 bits instead of 152 in the original text (additional 7 bits are used to distinguish between plain characters and repetition blocks).
          Given a text chunk, find its compressed representation which needs fewest number of bits to encode

    Input

          Input file contains a text chunk t. Its length doesn’t exceed 4096. A text chunk contains only small letters of the English alphabet.

    Output

          Print the length of the compressed text in bits at the first line of the output file. Print the compressed chunk itself at the second line of the output file. Use characters themselves to denote plain characters and “(r,l)” notation (without spaces) to denote repetition blocks.

    Sample Input

    aaabbaaabababababab 

    Sample Output

    95 
    aaabb(5,4)(2,10)

    题意概述

    给定一种压缩规则:或直接复制,或将重复出现的字符用二元数对(r,l)表示,代表[i − r ... i − r + l − 1]范围内的字符串。其中直接复制的字符占9位,每个二元数对占25位。对于给定的待压缩字符串,要求输出压缩后的最小数据位数和压缩后的编码。

    分析

    容易看出,字符串中前i位如何编码,对后继字符串的编码是没有影响的。这样,我们就可以想到运用保存决策的动态规划进行编码。 这里需要的一点小trick在于如何寻找尽量靠前的一段最长重复子串。代入kmp算法的思路,我们可以对原字符串的每一段后缀预处理出next数组,这样原字符串中每个前缀的最长重复子串就可以在O(n)的时间内得到了。于是这里可以得出一个复杂度O(n^2)的动态规划: 记minbit[i]为压缩前i个字符所需的最小数据量,str[i]为此时的决策,r[i],l[i]为这次转移时压缩成的数对(若不需压缩,str[i]=i, r[i]储存不需压缩的部分的起始点) 那么,每次转移时就可以从0到i-1枚举决策j,选择可以使(minbit[j] + (i-j)*9)或minbit[i - next[j][i-1]] + 25取得最小值的j,并保存决策。最后可以通过一个递归函数推出最终答案。 (这里需要注意一点问题……next数组是二维的,数据范围maxn为2^12,而内存限制为64000KB……如果用next[maxn][maxn]的方式开数组明显会MLE……所以我鼓起勇气采用了动态内存分配……)

     AC代码

     1 //Verdict: Accepted

     2 // Submission Date: 2014-10-02 16:34:32
     3 // Time: 8256MS
     4 // Memory: 34624KB
     5
     6 /*=============================================================================================================================*/
     7 /*======================================================Code by Asm.Def========================================================*/
     8 /*=============================================================================================================================*/
     9 #include <cstdio>
    10 #include <iostream>
    11 #include <algorithm>
    12 #include <cmath>
    13 #include <cctype>
    14 #include <memory.h>
    15 #include <cstring>
    16 #include <cstdlib>
    17 using namespace std;
    18 #define maxn ((int)4.1e3)
    19 /*===========================================================TYPES=============================================================*/
    20 typedef long long LL;
    21   
    22 /*======================================================GLOBAL VARIABLES=======================================================*/
    23 char ch[maxn];
    24 int len = 0, minbit[maxn], *next[maxn];
    25 int l[maxn], r[maxn], str[maxn];
    26 /*==========================================================FUNCTIONS==========================================================*/
    27 inline void getnext(int l){
    28     int i, j, L = len - l;
    29     next[l] = new int[L+2];
    30     int *Next = next[l];
    31     Next[0] = 0;
    32     for(i = 1;i < L;++i){
    33         j = Next[i-1] - 1;
    34         while(ch[l+i] != ch[l+j+1] && j >= 0)
    35             j = Next[j] - 1;
    36         if(ch[l+i] == ch[l+j+1])
    37             Next[i] = j + 2;
    38         else Next[i] = 0;
    39     }
    40 }
    41 void printpro(int i){
    42     if(str[i] == i){
    43         if(r[i])printpro(r[i]-1);
    44         int j;
    45         for(j = r[i];j <= i;++j)putchar(ch[j]);
    46         return;
    47     }
    48     printpro(str[i]);
    49     printf("(%d,%d)", r[i], l[i]);
    50 }
    51 int main(){
    52     #ifdef DEBUG
    53     assert(freopen("test","r",stdin));
    54     #endif
    55     //--------------------------------------------------variables-----------------------------------------------------------
    56       
    57     //-----------------------------------------------------work-------------------------------------------------------------
    58     char c;
    59     while(isalpha(c = getchar()))str[len] = len, ch[len++] = c;
    60     int i, j, Min, t;
    61     for(i = 0;i < len - 1; ++i)
    62         getnext(i);
    63     minbit[0] = 9;
    64     for(i = 1;i < len; ++i){
    65         Min = 0x7fffffff;
    66         for(j = 0;j < i;++j)
    67             if(minbit[j] + (i-j)*9 < Min){
    68                 Min = minbit[j] + (i-j)*9;
    69                 str[i] = i;
    70                 r[i] = j+1;
    71             }
    72         for(j = 0;j < i; ++j){
    73             t = next[j][i-j];
    74             if(!t)continue;
    75             if(minbit[i-t] + 25 < Min){
    76                 Min = minbit[i-t] + 25;
    77                 str[i] = i-t;
    78                 r[i] = i+1-t-j;
    79                 l[i] = t;
    80             }
    81         }
    82         minbit[i] = Min;
    83     }
    84     printf("%d ", minbit[len-1]);
    85     printpro(len-1);
    86     return 0;
    87 }
    88 /*=============================================================================================================================*/

  • 相关阅读:
    KindEditor简单的Demo使用
    动态从数据库读取菜单(ASP.NET版)
    完全卸载sql2005
    关于在xp(sp3 专业版)下安装sql2005开发版图解
    新手上路Tomcat 7.x和JDK的配置
    关于IE6幽灵字体
    【译】写个好的 CLI 程序
    【译】通过 Rust 学习解析器组合器 — Part 1
    【译】PHP 内核 — zval 基础结构
    【译】PHP 内核 — 字符串管理
  • 原文地址:https://www.cnblogs.com/Asm-Definer/p/4004337.html
Copyright © 2011-2022 走看看