这是一个很长的,所以请紧紧抓住。
这个想法是,当引用 Core.dll(见下文)时,应用程序将创建一个类型为的新对象Core
,并且在这样做时,构造函数将扫描C:\Problems
文件夹以查找包含使用该IPlugin
接口的类型的任何 DLL。
当我从控制台应用程序引用 Core.dll 时,它可以工作。当我从 PowerShell 执行等效操作时,它不起作用,除非 Core.dll 在 GAC 中。
这些是详细信息:
我有一个程序集(Core.dll),它是以下结果:
public class Core
{
List<IPlugin> plugins = new List<IPlugin>();
public Core(string path)
{
LoadPlugins(path);
foreach (IPlugin plugin in plugins)
plugin.Work();
}
void LoadPlugins(string path)
{
foreach (string file in Directory.GetFiles(path))
{
FileInfo fileInfo = new FileInfo(file);
Assembly dllAssembly = Assembly.LoadFile(file);
foreach (Type type in dllAssembly.GetTypes())
{
Type typeInterface = type.GetInterface("Problems.IPlugin");
if (typeInterface != null)
plugins.Add((IPlugin)Activator.CreateInstance(
dllAssembly.GetType(type.ToString())));
}
}
}
}
public interface IPlugin
{
string Name {get; set;}
void Work();
}
然后,我有一个单独的程序集 MyPlugin.dll,其中包含以下内容:
public class MyPlugin : IPlugin
{
public string Name { get; set; }
public void Work()
{
Console.WriteLine("Hello, World. I'm a plugin!");
}
}
我用第三个项目 TestApp.exe 测试了这一切(以 Core.dll 作为参考):
static void Main(string[] args)
{
Core core = new Core(@"C:\Problems");
Console.ReadLine();
}
我将 Core.dll 和 MyPlugin.dll 都放入 C:\Problems,TestApp.exe 就像一个魅力!因此,我编写了这个 PowerShell 脚本:
Add-Type -Path "C:\Problems\Core.dll"
$core = New-Object Problems.Core "C:\Problems"
这是我的结果(想象一堆红色):
New-Object : Exception calling ".ctor" with "1" argument(s): "Unable to load one or more o
f the requested types. Retrieve the LoaderExceptions property for more information."
At line:1 char:11
+ New-Object <<<< Problems.Core "C:\Problems"
+ CategoryInfo : InvalidOperation: (:) [New-Object], MethodInvocationException
+ FullyQualifiedErrorId : ConstructorInvokedThrowException,Microsoft.PowerShell.Commands.NewObjectCommand
深入研究异常,我发现:
PS C:\Problems> $Error[0].Exception.InnerException.LoaderExceptions
Could not load file or assembly 'Core, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null' or one of its dependencies. The system cannot find the file specified.
到目前为止,我的结论是,当我将 GetTypes() 作为插件加载器的一部分运行时,它看到 MyPlugin 继承自 Problems.IPlugin,它不知道在哪里找到它。但是,我不是已经在 Core.dll 中加载它了吗?
我可以解决这个问题的唯一方法是签署 Core.dll 程序集并将其添加到 GAC,但这既烦人又不符合我的目的。
(所有类都在 Problems 命名空间中,我省略了大量的清理和过滤逻辑来说明问题。)