2016-03-01 351 views
0

我想解压一个GZipped字符串,它是web服务响应的一部分。我有字符串是:解压缩GZIP流

"[31,-117,8,0,0,0,0,0,0,0,109,-114,65,11,-62,48,12,-123,-1,75,-50,-61,-42,-127,30,122,21,111,-126,94,60,-119,-108,-72,102,44,-48,-75,-93,-21,100,56,-6,-33,-19,20,20,101,57,37,95,-14,94,-34,4,-63,-5,-72,-73,-44,-110,-117,-96,38,-88,26,-74,38,-112,3,117,-7,25,-82,5,24,-116,56,-97,-44,108,-23,28,24,-44,-85,83,34,-41,97,-88,24,-99,23,36,124,-120,94,99,-120,15,-42,-91,-108,91,45,-11,70,119,60,-110,21,-20,12,-115,-94,111,-80,-93,89,-41,-65,-127,-82,76,41,51,-19,52,90,-5,69,-85,76,-96,-128,64,22,35,-33,-23,-124,-79,-55,-1,-2,-10,-87,0,55,-76,55,10,-57,122,-9,73,42,-45,98,-44,5,-77,101,-3,58,-91,39,38,51,-15,121,21,1,0,0]"

我尝试使用下面的方法来解串:

public static string UnZip(string value) 
     { 
      // Removing brackets from string 
      value = value.TrimStart('['); 
      value = value.TrimEnd(']'); 

      //Transform string into byte[] 
      string[] strArray = value.Split(','); 
      byte[] byteArray = new byte[strArray.Length]; 
      for (int i = 0; i < strArray.Length; i++) 
      { 
       if (strArray[i][0] != '-') 
        byteArray[i] = Convert.ToByte(strArray[i]); 
       else 
       { 
        int val = Convert.ToInt16(strArray[i]); 
        byteArray[i] = (byte)(val + 256); 
       } 
      } 

      //Prepare for decompress 
      System.IO.MemoryStream ms = new System.IO.MemoryStream(byteArray); 
      System.IO.Compression.GZipStream sr = new System.IO.Compression.GZipStream(ms, 
       System.IO.Compression.CompressionMode.Decompress); 

      //Reset variable to collect uncompressed result 
      byteArray = new byte[byteArray.Length]; 

      //Decompress 
      int rByte = sr.Read(byteArray, 0, byteArray.Length); 

      //Transform byte[] unzip data to string 
      System.Text.StringBuilder sB = new System.Text.StringBuilder(rByte); 
      //Read the number of bytes GZipStream red and do not a for each bytes in 
      //resultByteArray; 
      for (int i = 0; i < rByte; i++) 
      { 
       sB.Append((char)byteArray[i]); 
      } 
      sr.Close(); 
      ms.Close(); 
      sr.Dispose(); 
      ms.Dispose(); 
      return sB.ToString(); 
     } 

的方法是一个在下面的链接的修改版本: http://www.codeproject.com/Articles/27203/GZipStream-Compress-Decompress-a-string

不幸的是,该方法的结果是一个损坏的字符串。更具体地讲,我知道,输入字符串包含压缩的JSON对象,并输出字符串只有一些预期字符串:

"{\"rootElement\":{\"children\":[{\"children\":[],\"data\":{\"fileUri\":\"file:////Luciano/e/orto_artzi_2006_0_5_pixel/index/shapefiles/index_cd20/shp_all/index_cd2.shp\",\"relativePath\":\"/i" 

任何想法可能是什么问题,该如何解决呢?

+1

未压缩的结果可能会比压缩数据大。 'new byte [byteArray.Length]'只有在压缩比为1:1的情况下才有效。 – Luaan

回答

2

尝试

public static string UnZip(string value) 
{ 
    // Removing brackets from string 
    value = value.TrimStart('['); 
    value = value.TrimEnd(']'); 

    //Transform string into byte[] 
    string[] strArray = value.Split(','); 
    byte[] byteArray = new byte[strArray.Length]; 
    for (int i = 0; i < strArray.Length; i++) 
    { 
     byteArray[i] = unchecked((byte)Convert.ToSByte(strArray[i])); 
    } 

    //Prepare for decompress 
    using (System.IO.MemoryStream output = new System.IO.MemoryStream()) 
    { 
     using (System.IO.MemoryStream ms = new System.IO.MemoryStream(byteArray)) 
     using (System.IO.Compression.GZipStream sr = new System.IO.Compression.GZipStream(ms, System.IO.Compression.CompressionMode.Decompress)) 
     { 
      sr.CopyTo(output); 
     } 

     string str = Encoding.UTF8.GetString(output.GetBuffer(), 0, (int)output.Length); 
     return str; 
    } 
} 

MemoryBuffer()不“重复”的byteArray但它直接支持,所以你不能重用byteArray

我会补充一点,我觉得很有趣,他们将一个277个字符的json“压缩”为一个620个字符的字符串化字节数组。

作为一个旁注,这种方法的内存占用是超出了屋顶... 620字符串(实际上是一个277字节数组)要解压导致创建字符串/数组总大小为4887个字节(包括620个初始字符串)(免责声明:在执行该方法期间,GC可以回收这部分内存)。对于277字节的字节数组来说这是可以的......但对于更大的内存占用将变得相当大。