プロジェクトをNinject IoC
でセットアップしました。
私のプロジェクトには、通常のAsp.Net MVC
コントローラーとWeb Api
コントローラーがあります。現在、Ninject
はWeb Api
で動作しますが、Ninject
は通常のAsp.MVC
コントローラーでは動作しません。
My 通常のMVCコントローラー実装;
public class GalleryController : BaseController
{
public GalleryController(IUow uow)
{
Uow = uow;
}
........
}
通常のコントローラーで使用する場合のエラー
An error occurred when trying to create a controller of type 'Web.Controllers.HomeController'. Make sure that the controller has a parameterless public constructor.]
ただし、Web Api、で同じコードを試してみると動作します
public class GalleryController : BaseApiController
{
public GalleryController(IUow uow)
{
Uow = uow;
}
......
}
異なるリポジトリーを保持するインターフェース(ファクトリー・パターン)
public interface IUow
{
// Save pending changes to the data store.
void Commit();
//Repositoryries
IRepository<Gallery> Gallery { get; }
IMenuRepository Menus { get; }
}
NinjectDependencyScope
クラス;
public class NinjectDependencyScope : IDependencyScope
{
private IResolutionRoot resolver;
internal NinjectDependencyScope(IResolutionRoot resolver)
{
Contract.Assert(resolver != null);
this.resolver = resolver;
}
public void Dispose()
{
var disposable = resolver as IDisposable;
if (disposable != null)
disposable.Dispose();
resolver = null;
}
public object GetService(Type serviceType)
{
if (resolver == null)
throw new ObjectDisposedException("this", "This scope has already been disposed");
return resolver.TryGet(serviceType);
}
public IEnumerable<object> GetServices(Type serviceType)
{
if (resolver == null)
throw new ObjectDisposedException("this", "This scope has already been disposed");
return resolver.GetAll(serviceType);
}
}
NinjectDependencyResolver
クラス;
public class NinjectDependencyResolver : NinjectDependencyScope, IDependencyResolver
{
private IKernel kernel;
public NinjectDependencyResolver(IKernel kernel)
: base(kernel)
{
this.kernel = kernel;
}
public IDependencyScope BeginScope()
{
return new NinjectDependencyScope(kernel.BeginBlock());
}
}
Global.asaxのNinject
設定。
public class IocConfig
{
public static void RegisterIoc(HttpConfiguration config)
{
var kernel = new StandardKernel(); // Ninject IoC
//kernel.Load(Assembly.GetExecutingAssembly()); //only required for asp.net mvc (not for webapi)
// These registrations are "per instance request".
// See http://blog.bobcravens.com/2010/03/ninject-life-cycle-management-or-scoping/
kernel.Bind<RepositoryFactories>().To<RepositoryFactories>()
.InSingletonScope();
kernel.Bind<IRepositoryProvider>().To<RepositoryProvider>();
kernel.Bind<IUow>().To<Uow>();
// Tell WebApi how to use our Ninject IoC
config.DependencyResolver = new NinjectDependencyResolver(kernel);
}
}
Global.asax
protected void Application_Start()
{
// Tell WebApi to use our custom Ioc (Ninject)
IocConfig.RegisterIoc(GlobalConfiguration.Configuration);
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes);
BundleConfig.RegisterBundles(BundleTable.Bundles);
GlobalConfig.CustomizeConfig(GlobalConfiguration.Configuration);
AreaRegistration.RegisterAllAreas();
}
たくさん検索した結果、Web APIと通常のmvcでNinjectを使用できないことがわかりました。つまり、リポジトリを個別に構成する必要があります。
それから、asp.net mvc&web apiでNinjectを使用する方法を説明するニースの記事を見つけました: http://www.codeproject.com/Articles/412383/Dependency-Injection-in-asp-net- mvc4-and-webapi-us
そして今、私はエラーを取得せず、それが動作しています:D
更新1:
NinjectをMVCとWeb Apiで構成するための要点をいくつか書きました。ファイルを含めるだけです:
具象型のバインディングを追加するには、MainModuleのLoad()
メソッドに配置します。バインドを整理するために、好きなだけモジュールを作成できます。ただし、Modules
プロパティで返される配列に追加する必要もあります。
次にApplication_Start()
メソッドに追加します
NinjectContainer.RegisterModules(NinjectModules.Modules)
(MVCの場合)NinjectHttpContainer.RegisterModules(NinjectHttpModules.Modules)
(WebApiの場合)MVCとWebApiの両方の登録に同じ_NinjectModules.Modules
_を使用できることに注意してください。明確にするために分離しました
[〜#〜] update [〜#〜]:ランタイムに新しいカーネルをロードしてブートストラップするため、プロジェクトからNinjectWebCommon.csを忘れずに削除してください。残念ながらMVC専用です。
[〜#〜] update [〜#〜]:使用することもできます
NinjectContainer.RegisterAssembly()
(MVCの場合)NinjectHttpContainer.RegisterAssembly()
(WebApiの場合)これにより、現在のアセンブリですべてのモジュールがスキャンされます。このようにして、モジュールをプロジェクトのどこにでも配置でき、登録されます
MVC 5とWeb API 2.2では、次のNuGetパッケージを含めることでこの問題を解決しました。
Ninject.MVC5
Ninject.Web.WebApi.WebHost
Web APIの場合これにより他のNinject依存関係がインストールされ、RegisterServices
からNinjectWebCommon.cs
。
ここに私のためにうまくいく簡単な解決策があります:
パッケージマネージャーコンソールで1つずつ実行します。
インストールパッケージNinject
インストールパッケージNinject.MVC5
NinjectDependencyResolver.csをIoCフォルダーに追加します。
using Ninject;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Http.Dependencies;
namespace DemoApp.IoC
{
public class NinjectDependencyResolver : IDependencyResolver, System.Web.Mvc.IDependencyResolver
{
private readonly IKernel kernel;
public NinjectDependencyResolver(IKernel kernel)
{
this.kernel = kernel;
}
public IDependencyScope BeginScope()
{
return this;
}
public object GetService(Type serviceType)
{
return kernel.TryGet(serviceType);
}
public IEnumerable<object> GetServices(Type serviceType)
{
return kernel.GetAll(serviceType);
}
public void Dispose() { } //it is not necessary to implement any dispose logic here
}
}
App_Start/NinjectWebCommon.csで次の変更を行います。
CreateKernelメソッドに次の行を追加します。
NinjectDependencyResolver ninjectResolver = new NinjectDependencyResolver(kernel); DependencyResolver.SetResolver(ninjectResolver); // MVC GlobalConfiguration.Configuration.DependencyResolver = ninjectResolver; // Web API
RegisterServicesメソッドに次のようなバインディングを追加します。
kernel.Bind <IHelloService>()。To <HelloService>();
これで、NinjectWebCommon.csは次のようになります。
[Assembly: WebActivatorEx.PreApplicationStartMethod(typeof(DemoApp.App_Start.NinjectWebCommon), "Start")]
[Assembly: WebActivatorEx.ApplicationShutdownMethodAttribute(typeof(DemoApp.App_Start.NinjectWebCommon), "Stop")]
namespace DemoApp.App_Start
{
using System;
using System.Web;
using Microsoft.Web.Infrastructure.DynamicModuleHelper;
using Ninject;
using Ninject.Web.Common;
using DemoApp.IoC;
using System.Web.Mvc;
using System.Web.Http;
using DemoApp.Config;
public static class NinjectWebCommon
{
private static readonly Bootstrapper bootstrapper = new Bootstrapper();
/// <summary>
/// Starts the application
/// </summary>
public static void Start()
{
DynamicModuleUtility.RegisterModule(typeof(OnePerRequestHttpModule));
DynamicModuleUtility.RegisterModule(typeof(NinjectHttpModule));
bootstrapper.Initialize(CreateKernel);
}
/// <summary>
/// Stops the application.
/// </summary>
public static void Stop()
{
bootstrapper.ShutDown();
}
/// <summary>
/// Creates the kernel that will manage your application.
/// </summary>
/// <returns>The created kernel.</returns>
private static IKernel CreateKernel()
{
var kernel = new StandardKernel();
try
{
kernel.Bind<Func<IKernel>>().ToMethod(ctx => () => new Bootstrapper().Kernel);
kernel.Bind<IHttpModule>().To<HttpApplicationInitializationHttpModule>();
RegisterServices(kernel);
NinjectDependencyResolver ninjectResolver = new NinjectDependencyResolver(kernel);
DependencyResolver.SetResolver(ninjectResolver); //MVC
GlobalConfiguration.Configuration.DependencyResolver = ninjectResolver; //Web API
return kernel;
}
catch
{
kernel.Dispose();
throw;
}
}
/// <summary>
/// Load your modules or register your services here!
/// </summary>
/// <param name="kernel">The kernel.</param>
private static void RegisterServices(IKernel kernel)
{
kernel.Bind<IHelloService>().To<HelloService>();
}
}
}
HomeController.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Mvc;
using DemoApp.Config;
namespace DemoApp.Controllers
{
public class HomeController : Controller
{
private IHelloService helloService;
public HomeController(IHelloService helloService)
{
this.helloService = helloService;
}
// GET: /Home/
public string Index()
{
return "home/index: " + helloService.GetMessage();
}
}
}
UserController.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Http;
using DemoApp.Config;
namespace DemoApp.Controllers
{
public class UserController : ApiController
{
private IHelloService helloService;
public UserController(IHelloService helloService)
{
this.helloService = helloService;
}
[HttpGet]
public string Data()
{
return "api/user/data: " + helloService.GetMessage();
}
}
}
IHelloService.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
namespace DemoApp.Config
{
public interface IHelloService
{
string GetMessage();
}
}
HelloService.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
namespace DemoApp.Config
{
public class HelloService : IHelloService
{
public string GetMessage()
{
return "Hi";
}
}
}
次に、ブラウザでいくつかのテストを行います。私にとっては:
以上です。
問題は、Ninjectを使用してコントローラーを構築(および依存関係を解決)するControllerFactoryを登録していないことです。まだ独自のControllerFactoryを実装しようとしましたか?こちらもご覧ください http://bubblogging.wordpress.com/2012/06/04/mvc-controller-factory-ninject/ 。
これには、Nenadによるよりエレガントなソリューションがあります。最初に、既存のインフラストラクチャと競合するソリューションをここで実装しようとしたため、3時間余分にかかりました。 別のスタックオーバーフローの質問に対する回答です 。他の人が私が失われた時間を節約するのを助けるために、ここでその答えを複製しています。
MVCとASP.NET Web APIの間で同じコンテナを共有する方法があります。両方のインターフェースを実装するだけです。
public class NinjectDependencyResolver : NinjectDependencyScope, IDependencyResolver, System.Web.Mvc.IDependencyResolver
{
private readonly IKernel kernel;
public NinjectDependencyResolver(IKernel kernel)
: base(kernel)
{
this.kernel = kernel;
}
public IDependencyScope BeginScope()
{
return new NinjectDependencyScope(this.kernel.BeginBlock());
}
}
解決策については、この記事を確認してください: MVCとWeb APIの間で依存関係リゾルバーを共有する簡単な方法