6

将不胜感激这里的任何帮助。

关于场景的简要说明 -

服务器上运行着一个 COM+(用 C# 编写)。该 COM 的任务是获取多页 tiff 文件的文件名、页码以及将其转换为 gif 文件图像的分辨率。使用代理从 Web 应用程序调用此 COM。网站获取转换后的图像并以请求的分辨率显示。对于打印 - 它发出 2 个请求 - 第一个用于显示分辨率,第二个用于全分辨率(使用 window.print() 打印)。

问题 -

一段时间后,服务器内存不足,图像没有显示在网站上。服务器需要定期重启。

错误

EventType clr20r3, P1 imageCOM.exe, P2 1.0.0.0, P3 4fd65854, P4 prod.web.imaging, P5 1.0.0.0, P6 4fd65853, P7 1a, P8 21, P9 system.outofmemoryexception, P10 NIL.

Here is the error(s) on the web server (these continuously appear every minute) ….

System.Net.WebException: Unable to connect to the remote server ---> System.Net.Sockets.SocketException: No connection could be made because the target machine actively refused it
   at System.Net.Sockets.Socket.DoConnect(EndPoint endPointSnapshot, SocketAddress socketAddress)
   at System.Net.Sockets.Socket.InternalConnect(EndPoint remoteEP)
   at System.Net.ServicePoint.ConnectSocketInternal(Boolean connectFailure, Socket s4, Socket s6, Socket& socket, IPAddress& address, ConnectSocketState state, IAsyncResult asyncResult, Int32 timeout, Exception& exception)
   --- End of inner exception stack trace ---

我无权访问生产服务器,但系统管理员发送的错误指出 OutOfMemory。

因此,假设内存泄漏并专注于它 - 我的发现到目前为止处理这种情况的经验有限 -

  • Perfmon - 我看到进程/私有字节正在增加,.Net CLR 内存/堆中的字节数也在增加。因此,我认为它是托管内存泄漏。不过我不确定。
  • CPU 使用率 - 从 8% 开始,仅在开始时上升到 80%。它回落并保持在 3% - 12% 之间,但有几次回落到 75%-85%。不知道这里发生了什么。
  • 我开始调试服务器 COM 以查看堆、gcroot 等。

    1. 随着每个请求的提出,堆中有 2 个对象的计数正在增加。1 个对象是保存图像数据的对象。第二个对象是事件处理程序,用于在对象 1(图像)在一定时间后过期时从缓存中删除它。
    2. 查看方法调用 - 两个对象都导致相同的方法。
    3. 代码实现明智 - 请求的图像被缓存(最多一定数量) - 我可以理解这是为了更好的性能。可能这就是为什么对对象的引用在堆中增加,如项目符号 1 中所述。

我知道我给出了非常模糊的描述,但我需要某种线索来检测服务器上的真正问题。

编辑:图像对象已被处理为

Bitmap retVal;

      using (MemoryStream buffer = new MemoryStream(doc.FileData, 0, doc.DocumentSize, false, false))
      {
        using (Bitmap original = new Bitmap(buffer))
        {
        //select the page to convert and
        //perform scaling - full resolution or the requested resolution.  
        }
      }

      using (MemoryStream buffer = new MemoryStream())
      {
        retVal.Save(buffer, ImageFormat.Gif);
        retVal.Dispose();
        return buffer.GetBuffer();
      }
4

2 回答 2

2

确保在使用完Image对象(例如Bitmap)后对其进行处理。我猜您正在将 tiff 图像作为位图打开并重新缩放,然后再次将其保存为 gif。如果您不处理位图,它将泄漏内存(通常每次几 MB,具体取决于图像的大小)。

于 2012-08-23T15:57:51.360 回答
1

我认为您的第一点可能是问题的原因。您可以尝试减少缓存过期时间并检查再次出现内存不足错误所花费的时间,您可能必须更改算法以避免这种情况。你的第二个不应该是错误的原因。

于 2012-08-23T15:52:30.093 回答