5

我在使用 MemoryStream 时遇到了一些问题

我们正在使用第 3 方电子表格组件(类似于 excel),我正在尝试通过以下SaveHtml方法将数据保存为他们支持的 html。根据他们的文档,看起来非常简单。这是我的代码:

using (var memoryStream = new MemoryStream())
{
   this.ActiveSheet.SaveHtml(memoryStream);

   memoryStream.Position = 0;

   using (var streamReader = new StreamReader(memoryStream))
   {
       var htmlData = streamReader.ReadToEnd();
   }
}

设置时出现异常memoryStream.Position = 0

System.ObjectDisposedException: Cannot access a closed Stream.

快速浏览一下他们在 Reflector 中的 SaveHtml 方法会显示以下相关行:

public void SaveHtml(Stream stream)
{
    StreamWriter writer = null;

    try
    {
        writer = new StreamWriter(stream) { AutoFlush = true };

        writer.Write(str);
    }   
    finally
    {
        if (writer != null)
        {
            writer.Close();
        }
    }
}

我猜是因为 streamWriter 被他们的代码关闭了,我们不走运。关闭streamWriter,也就关闭了底层的Stream,对吧?

有什么办法可以解决这个问题?

谢谢。

4

2 回答 2

6

它似乎有效,因此您可以创建另一个内存流。提取缓冲区时不复制任何字节。

using (var memoryStream = new MemoryStream()) {
    this.ActiveSheet.SaveHtml(memoryStream);
    var buffer = memoryStream.GetBuffer();
    using (var memoryStream2 = new MemoryStream(buffer))
    using (var streamReader = new StreamReader(memoryStream2)) {
        var htmlData = streamReader.ReadToEnd();
    }
}

或者您可以推出自己的不可关闭的 MemoryStream 并将其提供给 SaveHtml。这也将阻止处理它,因为 Dispose 只是调用 Close。再次,恕我直言:

class MemoryStream2 : MemoryStream {
    public override void Close() { }
}
于 2012-07-20T21:00:33.460 回答
0

在这种情况下可以使用替代的内存流实现:

var stream = new ChunkedStream(pool, asOutputStreamOnDispose: true);

using (var writer = new StreamWriter(stream))
{
    writer.Write("hello world");
}

Assert.AreEqual(0, stream.Position);
Assert.AreEqual(ChunkedStreamState.ReadForward, stream.State);
Assert.AreEqual(3, pool.TotalAllocated);

using (var reader = new StreamReader(stream))
{
    Assert.AreEqual("hello world", reader.ReadToEnd());
    Assert.AreEqual(0, pool.TotalAllocated);
}

Assert.AreEqual(ChunkedStreamState.Closed, stream.State);
于 2017-07-30T12:31:29.820 回答