產生Controller
物件相關物件關係如下面UML圖
透過ControllerFactory
建立一個Controller
控制器物件.而ControllerFactory
依賴IControllerActivator
物件產生Controller
.
上面IControllerActivator
可以透過建立使用我們的依賴注入容器來替換原本反射產生物件.
DependencyResolver
是MVC提供的一個可替換物注入點,今天我們會藉由他來我們實現注入MVC方式.
在實現自己的DependencyResolver
前先談談Autofac容器做甚麼用的?
我之前有寫一篇IOC(控制反轉),DI(依賴注入) 深入淺出~~
講述 IOC(控制反轉),DI(依賴注入) 這兩個設計技巧的理念核心.
言簡意賅可以統一交由容器來幫忙管理物件生命週期和建立方式,也管理物件相依性,兩個重點我們使得只需要提供使用類別的特徵(型別或其他可辨別特徵),容器就提供給我們相對應的物件.
在Autofac
有需多使用方式這裡就不一一介紹,有興趣讀者可以上網google
或是查閱Autofac
官方文件
DependencyResolver
是一個靜態物件,MVC application使用同一個解析器(DefaultDependencyResolver
)而他有一個SetResolver
方法可以替換成其他DependencyResolver
IDependencyResolver
有兩個方法需要實現.
public interface IDependencyResolver
{
object GetService(Type serviceType);
IEnumerable<object> GetServices(Type serviceType);
}
MVC依賴於GetService
和GetServices
,取得物件實例並提供一個抽象提供外部提供修改或擴充.
預設使用(DefaultDependencyResolver
)這個解析器來取得我們物件(DefaultDependencyResolver
解析器使用Activator.CreateInstance(serviceType);
建立物件)
這邊我們利用autofac
來完成建立物件動作,先建立一個ILifetimeScope _container
由建構子注入此物件.
public class CustomerDependencyResolver : IDependencyResolver
{
private readonly ILifetimeScope _container;
public CustomerDependencyResolver(ILifetimeScope container)
{
if (container == null)
throw new ArgumentNullException(nameof (container));
_container = container;
}
public object GetService(Type serviceType)
{
return _container.ResolveOptional(serviceType);
}
public IEnumerable<object> GetServices(Type serviceType)
{
return (IEnumerable<object>) _container.ResolveOptional(typeof (IEnumerable<>).MakeGenericType(serviceType));
}
}
在GetService
呼叫ResolveOptional
方法透過Type
到容器中搜尋匹配的物件並返回.
IControllerActivator
有一個Create
方法,ControllerFacotry
靠它來幫我們產生使用Controller
物件,而我們在這邊建立自己IControllerActivator
並在Create
方法中實現自己得邏輯.透過DependencyResolver
來產生物件(替換成CustomerDependencyResolver
)
public class CustomerControllerActivator : IControllerActivator
{
public IController Create(RequestContext requestContext, Type controllerType)
{
return (IController) DependencyResolver.Current.GetService(controllerType);
}
}
我們會在Autofac
容器註冊目前Assembly
所有繼承IController
物件.
//注入typeof(MvcApplication).Assembly 中所有繼承IController物件.
builder.RegisterControllers(typeof(MvcApplication).Assembly);
在上面CustomerControllerActivator.Create
會透Autofac
解析器幫我們建立Controller
首先利用ControllerBuilder
的SetControllerFactory
方法,重新替換使用ControllerFacotry
.
在利用builder.RegisterControllers
注入typeof(MvcApplication).Assembly
中所有繼承IController
物件.
註冊IMemberService
介面物件(裡面有一個int GetMemberBalance(int memberId);
方法來模擬取得會員餘額)
DependencyResolver.SetResolver(new CustomerDependencyResolver(builder.Build()))
替換成我們使用的解析器
因為
ControllerFacotry
預設使用DefaultControllerActivator
,而我們需要替換成自己建立得CustomerControllerActivator
並利用容器來幫我們注入.
public class MvcApplication : HttpApplication
{
protected void Application_Start()
{
RouteConfig.RegisterRoutes(RouteTable.Routes);
BundleConfig.RegisterBundles(BundleTable.Bundles);
//把DefaultControllerFactory 中的IControllerActivator替換成我們自己寫的CustomerControllerActivator
ControllerBuilder.Current.SetControllerFactory(
new DefaultControllerFactory(new CustomerControllerActivator()));
AutofacRegister();
}
private static void AutofacRegister()
{
ContainerBuilder builder = new ContainerBuilder();
//注入typeof(MvcApplication).Assembly 中所有繼承IController物件.
builder.RegisterControllers(typeof(MvcApplication).Assembly);
builder.RegisterType<MemberService>().As<IMemberService>();
//替換成自己的DependencyResolver
DependencyResolver.SetResolver(new CustomerDependencyResolver(builder.Build()));
}
}
HomeController
控制器中在建構子注入,並呼叫IMemberService.GetMemberBalance
方法
執行專案請求Home/About
頁面可以看到ViewBag.Message
已經成功顯示一個HardCode餘額了.
public class HomeController : Controller
{
private readonly IMemberService _service;
public HomeController(IMemberService service)
{
_service = service;
}
public ActionResult About()
{
ViewBag.Message = $"Member Balance { _service.GetMemberBalance(123)}";
return View();
}
}
DefaultControllerActivator
使用反射建立一個Controller
物件
然而IControllerActivator
提供一個產生Controller
接口,而我們可以藉由實現此介面並使用DependencyResolver
靜態物件產生Controller
物件(藉由容器框架產生).
最後會把Controller
依賴物件藉由依賴注入容器注入進去.
Github範例程式原始碼 CustomerContainer
分支上