3

我正在学习如何将 nInject 用于我正在开发的新应用程序,并且我创建了以下示例代码,可以将其复制/粘贴到一个简单的控制台应用程序中。它成功返回了 IFoo 的一个实例,但我对此有疑问。

我将如何修改代码以让 FooManager 类创建 Foo 对象的实例而不执行“新建”。内核是否也必须注入?但是,如果内核被注入并且我将行更改为 read var foo = _kernel.Get<IFoo>(),那不是引入了服务定位器反模式吗?

namespace IOCTest
{
    class Program
    {
        static void Main(string[] args)
        {
            using (IKernel kernel = new StandardKernel(new StandardModule()))
            {
                // do something with the kernal
                var mgr = kernel.Get<IFooManager>();
                var foo = mgr.GetById(1);
            }
        }
    }

    public class StandardModule : Ninject.Modules.NinjectModule
    {
        public override void Load()
        {
            Bind<IDatabase>()
                .To<Database>()
                .InTransientScope();

            Bind<IFooManager>()
                .To<FooManager>()
                .InTransientScope();
        }
    }

    //******************************************************

    public interface IDatabase
    {
        object[] GetScalar(int id);
    }

    public class Database : IDatabase
    {
        public object[] GetScalar(int id)
        {
            return new object[] { "RowName" };
        }
    }

    //******************************************************

    public interface IFooManager
    {
        IFoo GetById(int id);
    }

    public class FooManager : IFooManager
    {
        private IDatabase _db;

        public FooManager(IDatabase db) { _db = db; }

        public IFoo GetById(int id)
        {
            var results = _db.GetScalar(id);
            var foo = new Foo();   // <-- HOW DO I ELIMINATE THIS DEPENDENCY?
            foo.Name = results[0].ToString();
            return foo;
        }
    }

    //******************************************************

    public interface IFoo
    {
        string Name { get; set; }
    }

    public class Foo : IFoo
    {
        public string Name { get; set; }
    }

    //******************************************************
}
4

2 回答 2

4

首先你必须考虑 Foo 的目的。这是某种数据容器还是某种服务?

在第一种情况下,您的代码就像它一样完美。数据容器没有依赖关系,不应由 IoC 容器创建。

在第二种情况下,阅读有关 Ninject.Extensions.Factory 的信息。

http://www.planetgeek.ch/2011/12/31/ninject-extensions-factory-introduction/

https://github.com/ninject/ninject.extensions.factory/wiki

于 2012-04-10T18:49:08.147 回答
2

有几种方法可以消除这种依赖。您可以执行与数据库依赖项相同的操作并使用构造函数注入。您可以进行属性注入(https://github.com/ninject/ninject/wiki/Injection-Patterns)。另一种方式,也许是您正在寻找的,将是服务位置。为此,您可以更新 FooManager ctor 以要求 IKernel。这将自动解决,然后您可以使用传入的内核来获取 Foo.

public class FooManager : IFooManager
{
    private IDatabase _db;
    private IKernel _kernel;

    public FooManager(IDatabase db, IKernel kernel) { _db = db; _kernel = kernel;}

    public IFoo GetById(int id)
    {
        var results = _db.GetScalar(id);
        // var foo = new Foo();   // <-- HOW DO I ELIMINATE THIS DEPENDENCY?
        var foo = kernel.Get<IFoo>(); // Like this perhaps
        foo.Name = results[0].ToString();
        return foo;
    }
}
于 2012-04-10T15:42:52.327 回答