我有一些 C# 代码使用 CSharpCodeProvider.CompileAssemblyFromSource 在内存中创建程序集。程序集被垃圾回收后,我的应用程序使用的内存比创建程序集之前多。我的代码在 ASP.NET 网络应用程序中,但我在 WinForm 中复制了这个问题。我正在使用 System.GC.GetTotalMemory(true) 和 Red Gate ANTS Memory Profiler 来测量增长(示例代码大约 600 字节)。
从我所做的搜索来看,这听起来像是泄漏来自新类型的创建,而不是真正来 self 持有引用的任何对象。我找到的一些网页提到了有关 AppDomain 的内容,但我不明白。有人可以解释这里发生了什么以及如何解决它吗?
这里有一些泄漏的示例代码:
private void leak()
{
CSharpCodeProvider codeProvider = new CSharpCodeProvider();
CompilerParameters parameters = new CompilerParameters();
parameters.GenerateInMemory = true;
parameters.GenerateExecutable = false;
parameters.ReferencedAssemblies.Add("system.dll");
string sourceCode = "using System;\r\n";
sourceCode += "public class HelloWord {\r\n";
sourceCode += " public HelloWord() {\r\n";
sourceCode += " Console.WriteLine(\"hello world\");\r\n";
sourceCode += " }\r\n";
sourceCode += "}\r\n";
CompilerResults results = codeProvider.CompileAssemblyFromSource(parameters, sourceCode);
Assembly assembly = null;
if (!results.Errors.HasErrors)
{
assembly = results.CompiledAssembly;
}
}
更新1:这个问题可能是相关的:Dynamically loading and unloading a a dll generated using CSharpCodeProvider
更新 2: 为了进一步了解应用程序领域,我发现了这个:What is an application domain - an explanation for .Net beginners
更新 3:为了澄清,我正在寻找一种解决方案,它提供与上述代码相同的功能(编译并提供对生成代码的访问)而不会泄漏内存。看起来解决方案将涉及创建新的 AppDomain 和编码(marshal)处理。
最佳答案
我想我有一个可行的解决方案。感谢大家给我指出正确的方向(我希望如此)。
程序集不能直接卸载,但 AppDomains 可以。我创建了一个帮助程序库,它被加载到一个新的 AppDomain 中,并且能够从代码编译一个新的程序集。该帮助程序库中的类如下所示:
public class CompilerRunner : MarshalByRefObject
{
private Assembly assembly = null;
public void PrintDomain()
{
Console.WriteLine("Object is executing in AppDomain \"{0}\"",
AppDomain.CurrentDomain.FriendlyName);
}
public bool Compile(string code)
{
CSharpCodeProvider codeProvider = new CSharpCodeProvider();
CompilerParameters parameters = new CompilerParameters();
parameters.GenerateInMemory = true;
parameters.GenerateExecutable = false;
parameters.ReferencedAssemblies.Add("system.dll");
CompilerResults results = codeProvider.CompileAssemblyFromSource(parameters, code);
if (!results.Errors.HasErrors)
{
this.assembly = results.CompiledAssembly;
}
else
{
this.assembly = null;
}
return this.assembly != null;
}
public object Run(string typeName, string methodName, object[] args)
{
Type type = this.assembly.GetType(typeName);
return type.InvokeMember(methodName, BindingFlags.InvokeMethod, null, assembly, args);
}
}
它非常基础,但足以进行测试。 PrintDomain 用于验证它是否存在于我的新 AppDomain 中。编译获取一些源代码并尝试创建一个程序集。 Run 让我们测试从给定的源代码执行静态方法。
下面是我如何使用辅助库:
static void CreateCompileAndRun()
{
AppDomain domain = AppDomain.CreateDomain("MyDomain");
CompilerRunner cr = (CompilerRunner)domain.CreateInstanceFromAndUnwrap("CompilerRunner.dll", "AppDomainCompiler.CompilerRunner");
cr.Compile("public class Hello { public static string Say() { return \"hello\"; } }");
string result = (string)cr.Run("Hello", "Say", new object[0]);
AppDomain.Unload(domain);
}
它基本上创建域,创建我的帮助程序类 (CompilerRunner) 的实例,使用它来编译新程序集(隐藏),运行新程序集中的一些代码,然后卸载域以释放内存。
您会注意到 MarshalByRefObject 和 CreateInstanceFromAndUnwrap 的使用。这些对于确保帮助程序库确实存在于新域中非常重要。
如果有人注意到任何问题或有改进建议,我很乐意听取他们的意见。
关于c# - 如何防止 CompileAssemblyFromSource 泄漏内存?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1799373/