我一直试图诊断仅在Windows 7 / Server 2008 R2上出现的服务中的内存泄漏。我将其范围缩小到我们使用Microsoft.Web.Administration.ServerManager收集我们站点中应用程序信息的位置。我将其缩小到下面的控制台应用程序,它表现出相同的行为。它可能仍然比需要的复杂,但是我想尽可能地模仿服务的行为。
我发现先前的问题here非常相似,并做出了答案中建议的更改。这似乎降低了增长率,但仍然显着泄漏(在注释“原始测试”下,我已注释掉根据这些答案更改的代码。“修改的测试”注释表明了我所做的更改。最初有GC.Collect调用,当我在Windows 10系统上运行它时,它增长了一段时间,然后才启动垃圾收集。有了GC.Collect调用,它运行就没有在Win 10上增长,但在Win 7上没有任何区别。
我在探查器下运行它,它表明泄漏的内存是本机的,并且泄漏来自nativerd.dll。
有人遇到这样的问题吗?我是C#的新手,仍在学习垃圾回收的工作原理,所以我想知道我是否做错了什么?
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.Web.Administration;
namespace ServerManagerLeakTest
{
class Program
{
static void Main(string[] args)
{
Console.Write("Working.");
var me = new MyClass();
me.Run();
}
}
internal class MyClass
{
ServerManagerWrapper _smw = new ServerManagerWrapper();
public void Run()
{
while (true)
{
var t = Task.Run(async delegate
{
DoWork();
await Task.Delay(1000);
});
try
{
t.Wait();
}
catch (Exception e)
{
Console.Write("Main Exception: " + e.Message);
}
Console.Write(".");
}
}
public void DoWork()
{
try
{
var data = _smw.GetWebApps().ToList();
data.Clear();
}
catch (Exception e)
{
Console.Write("DoWork Exception: " + e.Message);
}
}
}
internal class ServerManagerWrapper
{
public List<int> GetWebApps()
{
List<int> result = new List<int>() { };
// Original Test
//
// using (var serverManager = new ServerManager())
// {
// foreach (var site in serverManager.Sites)
// {
// result.AddRange(GetWebApps(site));
// }
//}
// Modified Test
var serverManager = new ServerManager();
foreach (var site in serverManager.Sites)
{
result.AddRange(GetWebApps(site));
}
serverManager.Dispose();
serverManager = null;
System.GC.Collect();
return result;
}
private IEnumerable<int> GetWebApps(Site site)
{
// Original Test
//
//for (var application in site.Applications)
//{
// yield return application.GetHashCode();
//}
// Modified Test
List<int> result = new List<int>() { };
for (int i = 0; i < site.Applications.Count; i++)
{
result.Add(site.Applications[i].GetHashCode());
}
return result;
}
}
}
答案 0 :(得分:0)
@Lex Li的评论中提供了答案。
将支票移至另一个过程。调用IIS REST API,PowerShell甚至是appcmd并解析结果。让泄漏不由您自己承担。