我需要在加载dll后检查运行GetTypes()的时间量。 代码如下。
Assembly assem = Assembly.LoadFrom(file);
sw = Stopwatch.StartNew();
var types1 = assem.GetTypes();
sw.Stop();
double time1 = sw.Elapsed.TotalMilliseconds;
我想卸载并重新加载dll来检查再次运行GetTypes()的时间。
assem = null
足够好了吗? 答案 0 :(得分:68)
您可以使用其他AppDomain吗?
AppDomain dom = AppDomain.CreateDomain("some");
AssemblyName assemblyName = new AssemblyName();
assemblyName.CodeBase = pathToAssembly;
Assembly assembly = dom.Load(assemblyName);
Type [] types = assembly.GetTypes();
AppDomain.Unload(dom);
答案 1 :(得分:34)
不幸的是,加载后无法卸载程序集。但是您可以卸载AppDomain。您可以做的是创建一个新的AppDomain(AppDomain.CreateDomain(...)),将程序集加载到此appdomain中以使用它,然后在需要时卸载AppDomain。卸载AppDomain时,将卸载已加载的所有程序集。 (见reference)
要调用垃圾收集器,可以使用
GC.Collect(); // collects all unused memory
GC.WaitForPendingFinalizers(); // wait until GC has finished its work
GC.Collect();
GC在后台线程中调用终结器,这就是为什么你必须等待并再次调用Collect()以确保删除所有内容。
答案 2 :(得分:31)
您可以将LoadFrom()
与LoadFile()
一起使用,而不是使用Load
或File.ReadAllBytes()
。有了它,它不会使用汇编文件,但会读取它并使用读取数据。
您的代码将如下所示
Assembly assem = Assembly.Load(File.ReadAllBytes(filePath));
sw = Stopwatch.StartNew();
var types1 = assem.GetTypes();
sw.Stop();
double time1 = sw.Elapsed.TotalMilliseconds;
从here我们无法卸载该文件,除非卸载其中包含的所有域。
希望这会有所帮助。:)
答案 3 :(得分:4)
您无法从当前AppDomain卸载程序集。但是您可以创建新的AppDomain,将程序集加载到其中,在新的AppDomain中执行一些代码然后卸载它。请查看以下链接:MSDN
答案 4 :(得分:0)
不幸的是,无法卸载程序集,而且-如果您使用appdomains-那么它将阻止您与主应用程序的api /程序集通信。
关于问题的最佳描述可以在这里找到:
脚本托管指南 http://www.csscript.net/help/Script_hosting_guideline_.html
如果您想在不与主应用程序通信的情况下运行C#代码,那么最好的方法是集成C#脚本API:
对于集成,您将需要以下软件包:
C#脚本: http://www.csscript.net/CurrentRelease.html
Visual Studio扩展: https://marketplace.visualstudio.com/items?itemName=oleg-shilo.cs-script
但是,如果您想从C#脚本与应用程序进行通信-当时只能使用具有不断变化的程序集名称的同一个appDomain-但不幸的是,这会占用ram和磁盘空间。
如何完成代码示例-可在此处找到:
https://github.com/tapika/cppscriptcore CsScriptHotReload.sln
这是演示视频:
https://drive.google.com/open?id=1jOECJj0_UPNdllwF4GWb5OMybWPc0PUV
答案 5 :(得分:0)
如果只想加载初始程序集而没有任何依赖程序集,则可以在AppDomain中使用Assembly.LoadFile
,然后在完成后卸载AppDomain。
创建一个加载程序类以加载和使用程序集:
class Loader : MarshalByRefObject
{
public void Load(string file)
{
var assembly = Assembly.LoadFile(file);
// Do stuff with the assembly.
}
}
在单独的应用程序域中运行加载程序,如下所示:
var domain = AppDomain.CreateDomain(nameof(Loader), AppDomain.CurrentDomain.Evidence, new AppDomainSetup { ApplicationBase = Path.GetDirectoryName(typeof(Loader).Assembly.Location) });
try {
var loader = (Loader)domain.CreateInstanceAndUnwrap(typeof(Loader).Assembly.FullName, typeof(Loader).FullName);
loader.Load(myFile);
} finally {
AppDomain.Unload(domain);
}