使用属性注入而不是构造函数注入

时间:2011-02-02 02:00:26

标签: c# asp.net-mvc ninject elmah

长话短说,我正在尝试将ELMAH用于MVC 2和Ninject,我需要使用无参数构造函数。我在这里创建了一篇关于它的帖子:Using a parameterless controller constructor with Ninject?

我被建议使用属性注入而不是构造函数注入。所以我离开了这个:

public class DepartmentsController : Controller
{
    private IDepartmentsRepository departmentsRepository;

    public DepartmentsController(IDepartmentsRepository departmentsRepository)
    {
        this.departmentsRepository = departmentsRepository;
    }

    ...
}

到此:

public class DepartmentsController : Controller
{
    private IDepartmentsRepository _departmentsRepository;

    [Inject]
    public IDepartmentsRepository DepartmentsRepository
    {
        get { return _departmentsRepository; }
        set { _departmentsRepository = value; }
    }

    ...
}

但在我的其他控制器函数中,无论我是否尝试访问DepartmentsRepository或_departmentsRepository,当我尝试访问它时,我得到对象引用未设置为对象的实例错误。

我还需要做些什么吗?

5 个答案:

答案 0 :(得分:11)

我有类似的问题。看看我的问题:Using Ninject with Membership.Provider

基本上,当你初始化DepartmentsController时,你需要注入this(即你的部门控制器进入你的Ninject核心。所以它类似于:

public class DepartmentsController : Controller
{
  private IDepartmentsRepository _departmentsRepository;

  [Inject]
  public IDepartmentsRepository DepartmentsRepository
  {
    get { return _departmentsRepository; }
    set { _departmentsRepository = value; }
  }

  public DepartmentsController()
  {
    NinjectHelper.Kernel.Inject(this);
  }
}

在这种情况下,NinjectHelper获取当前的Ninject Kernel。

答案 1 :(得分:3)

尝试这样的事情:

的Global.asax.cs

        protected void Application_Start()
        {
            DependencyResolver.SetResolver(
                new MyDependencyResolver(
                    new StandardKernel(
                        new MyModule())));
            //...
        }

MyDependencyResolver.cs

    public class MyDependencyResolver : IDependencyResolver
    {
        private IKernel kernel;

        public MyDependencyResolver(IKernel kernel)
        {
            this.kernel = kernel;
        }

        public object GetService(Type serviceType)
        {
            return kernel.TryGet(serviceType);
        }

        public IEnumerable<object> GetServices(Type serviceType)
        {
            return kernel.GetAll(serviceType);
        }
    }

MyModule.cs

    public class MyModule : NinjectModule
    {
        public override void Load()
        {
            Bind<IDepartmentsRepository>().To<DepartmentsRepository>();
        }
    }

答案 2 :(得分:2)

对象引用未设置异常可能有两个原因。

1)Ninject不知道如何将IDepartmentsRepository绑定到DepartmentsRepository的具体实现(我怀疑是这种情况)

2)如果您尝试访问控制器构造函数中的DepartmentsRepository属性,它将抛出异常(因为Ninject只能在构造对象后注入Property Dependencies)。

希望有所帮助。

答案 3 :(得分:0)

正如Daniel T.在上述评论中所发表的那样,您应该查看Ninject.Web.Mvc。如果你在该项目中使用NinjectHttpApplication,它将为你自动装配一切,这样当NinjectControllerFactory构造一个新的控制器时,它会调用Inject()来填充属性注入。

答案 4 :(得分:0)

即使没有特别使用MVC控制器,到达Ninject仍遇到“使用属性注入而不是构造函数注入”问题的任何人的观察。

Ninject将仅标识属性的[Inject]属性,并对作为DI的Ninject链的一部分而变得生动活泼的类执行属性注入。

如果要创建这样的对象

TreeMap<Integer, List<String>> map = frequencies.entrySet().stream() .collect(Collectors.groupingBy( Map.Entry<String, Integer>::getValue, TreeMap::new, Collectors.toList() )); Map.Entry<Integer, List<String>> largest = map.lastEntry();

Ninject不知道类的实例化,因此也不知道执行任何注入。

在MVC世界中,您可以使用

var myObj = new MyObj();