c# - Ninject + 自动发现

标签 c# ninject inversion-of-control

事情是这样的:我想创建一个 C# 控制台应用程序。运行时,此应用程序将在特定文件夹中查找具有实现特定接口(interface)的类的 dll,然后在这些 dll 上运行方法。

我以前没有这样做过,但从我的阅读来看,从 IoC/Ninject 的角度来看,这应该“足够简单”。我认为您可以使用 kernel.Bind() 来加载特定目录中特定接口(interface)的程序集。我想/希望我能解决这个问题(如果您不知道,请告诉我!)。

但这就是我的困境。

这是一个首先可以提供帮助的视觉效果:

MainProgramFolder
-- MainProgram.exe
-- MainProgram.exe.config
-- LibraryFolder
----- Library1Folder
--------Library1.dll
--------Library1.dll.config
----- Library2Folder
--------Library2.dll
--------Library2.dll.config

实现此接口(interface)的 dll 在技术上是独立的应用程序 - 它们只是库而不是 exe(或者更确切地说,我希望它们用于 IoC 目的)。我希望它们能够在自己的上下文中运行,并使用自己的 app.configs。例如,MainProgram.exe 会将 ILibrary 接口(interface)绑定(bind)到 Library1.dll 和 Library2.dll 内的类,因为它们实现 ILibrary。但在 Library1 内部,它调用 ConfigurationManager 来获取其设置。当我为 MainProgram 中的每个绑定(bind)调用 Class.Method() 时,如何确保它们引用自己的 .config 而不是 MainProgram.exe.config? (此外,fwiw,这些附加库可能不是程序集的一部分,甚至不是主程序的命名空间的一部分——我们基本上为应用程序提供了一个放置文件夹,以“订阅”主程序的执行。)

IOW,我知道您可以将 app.config 附加到类库,但我不知道在从 IOC 解析绑定(bind)后如何使这些 dll“看到”自己的配置而不是主配置程序的配置。

感谢所有想法!

谢谢 汤姆

最佳答案

首先,要加载并绑定(bind)所有类,您需要 ninject.extensions.conventions 以及类似这样的内容:

        var kernel = new StandardKernel();
        /*add relevant loop/function here to make it recurse folders if need be*/
        kernel.Bind(s => s.FromAssembliesMatching("Library*.dll")
            .Select(type => type.IsClass && type.GetInterfaces().Contains(typeof(ILibrary)))
            .BindSingleInterface()
            .Configure(x=>x.InSingletonScope()));

要使每个实例加载其配置,就好像它是入口点一样,您需要在新的应用程序域中运行它。您的 ILibrary 实现需要继承 MarshalByRefObject 并可序列化,以便它能够在备用应用程序域中正确运行

[Serializable]
    public class LibraryA :MarshalByRefObject, ILibrary

然后,您可以将此激活策略添加到内核中,这将导致内核在返回之前将 ILibrary 实例与在备用应用程序域中加载的实例(按照您的配置文件约定)进行交换。

public class AlternateAppDomainStrategy<T> : ActivationStrategy
    {
        public override void Activate(IContext context, InstanceReference reference) 
        {
            if (reference.Instance.GetType().GetInterfaces().Contains(typeof(T)))
            {
                var type = reference.Instance.GetType();

                var configFilePath = type.Assembly.GetName().Name + ".dll.config";
                var file = new FileInfo(configFilePath);
                if (file.Exists)
                {
                    var setup = new AppDomainSetup() { ConfigurationFile = file.FullName, ApplicationBase = AppDomain.CurrentDomain.BaseDirectory };
                    var domain = AppDomain.CreateDomain(type.FullName, null, setup);

                    var instance = domain.CreateInstanceAndUnwrap(type.Assembly.FullName, type.FullName);

                    reference.Instance = instance;
                }
                else
                {
                    throw new FileNotFoundException("Missing config file", file.FullName);
                }
            }
        }
    }

并将其添加到您的内核

kernel.Components.Add<IActivationStrategy, AlternateAppDomainStrategy<ILibrary>>();

从那里您可以简单地实例化 ILibrary 实例并调用它们的方法。他们将使用自己的配置加载自己的应用程序域。如果您需要通过方法或构造函数将事物传入/传出实例,那么事情会变得更加复杂,但如果您不需要,那么从声音来看,这应该没问题。

var libs = kernel.GetAll<ILibrary>();
            foreach (var lib in libs)
            {
                lib.Method();
            }

关于c# - Ninject + 自动发现,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20007770/

相关文章:

c# - Visual Studios 2015 中的 "Restore NuGet Packages"未执行任何操作

c# - 如何对数百万位长的数字进行计算?

c# - 在 C# 中重写 protected 内部虚方法

c# - 我应该使用构造函数注入(inject)还是 IoC.Resolve?

node.js - 反转 : contextual injection with class name

c# - 使 Gridview 的一个字段成为选择链接?

c# - 强制 HttpWebRequest 使用 HTTP 1.0 并在 Windows 8 的 PCL 中设置 UserAgent

wcf - Ninject 和 WCF 服务授权管理器

Ninject 和 forwarded 类型(在 CaSTLe Windsor 中众所周知)

c# - Ninject.MVC5 不生成 NinjectWebCommon.Cs