gpt4 book ai didi

c# - GZipStream 正在切断 XML 的最后一部分

转载 作者:太空狗 更新时间:2023-10-30 00:46:18 24 4
gpt4 key购买 nike

我创建了一个名为 AddGZip 的扩展方法,如下所示:

public static void AddGZip(this HttpResponse response)
{
response.Filter = new GZipStream(response.Filter, CompressionMode.Compress);
response.AppendHeader("Content-Encoding", "gzip");
}

这是一个非常简化的代码版本:

var response = HttpContext.Current.Response;
var request = HttpContext.Current.Request;
var result = File.ReadAllText(path);
if (request.SupportsGZip)
{
response.AddGZip();
}
response.Write(result);
response.Flush();

当您在支持 GZip 的网络浏览器中查看响应时,您会收到如下错误:

"XML Parsing Error: unclosed token Location: http://webserver1/1234.xml Line Number 78, Column 1:"

当我查看源代码时,它基本上错过了 XML 文件末尾的最后一个 >。所以 1 或 2 个字节。

如果我注释掉 AddGZip 行,它工作正常。但是我真的很想支持 GZip,因为 XML 可能非常大。

有人对我有什么建议吗?我已经尝试检查很多博客,但似乎没有针对此类错误的解决方案。

戴夫

最佳答案

DeflateStream(GZipStream 建立在 DeflateStream 并继承了问题*),其中刷新可能会丢失数据。

Response.Flush() 将刷新过滤器。解决方案是使用一个包装器,它知道压缩和底层接收器,并且只刷新后者:

public enum CompressionType
{
Deflate,
GZip
}
/// <summary>
/// Provides GZip or Deflate compression, with further handling for the fact that
/// .NETs GZip and Deflate filters don't play nicely with chunked encoding (when
/// Response.Flush() is called or buffering is off.
/// </summary>
public class WebCompressionFilter : Stream
{
private Stream _compSink;
private Stream _finalSink;
public WebCompressionFilter(Stream stm, CompressionType comp)
{
switch(comp)
{
case CompressionType.Deflate:
_compSink = new DeflateStream((_finalSink = stm), CompressionMode.Compress);
break;
case CompressionType.GZip:
_compSink = new GZipStream((_finalSink = stm), CompressionMode.Compress);
break;
}
}
public override bool CanRead
{
get
{
return false;
}
}
public override bool CanSeek
{
get
{
return false;
}
}
public override bool CanWrite
{
get
{
return true;
}
}
public override long Length
{
get
{
throw new NotSupportedException();
}
}
public override long Position
{
get
{
throw new NotSupportedException();
}
set
{
throw new NotSupportedException();
}
}
public override void Flush()
{
//We do not flush the compression stream. At best this does nothing, at worse it
//loses a few bytes. We do however flush the underlying stream to send bytes down the
//wire.
_finalSink.Flush();
}
public override long Seek(long offset, SeekOrigin origin)
{
throw new NotSupportedException();
}
public override void SetLength(long value)
{
throw new NotSupportedException();
}
public override int Read(byte[] buffer, int offset, int count)
{
throw new NotSupportedException();
}
public override void Write(byte[] buffer, int offset, int count)
{
_compSink.Write(buffer, offset, count);
}
public override void WriteByte(byte value)
{
_compSink.WriteByte(value);
}
public override void Close()
{
_compSink.Close();
_finalSink.Close();
base.Close();
}
protected override void Dispose(bool disposing)
{
if(disposing)
{
_compSink.Dispose();
_finalSink.Dispose();
}
base.Dispose(disposing);
}
}

同样值得注意的是,大多数支持 gzip 编码的用户代理也支持 deflate 编码。虽然 deflate 的大小改进可以忽略不计(字面意思是几个字节),但某些架构上的一些库处理 deflate 的效果要好得多(这适用于压缩和解压缩),因此始终值得使用 HTTP 压缩的 gzip 来支持 deflate。

关于c# - GZipStream 正在切断 XML 的最后一部分,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3653250/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com