ASP.NET MVC 视图(三)
前言
上篇对于Razor视图引擎和视图的类型做了大概的讲解,想必大家对视图的本身也有所了解,本篇将利用IoC框架对视图的实现进行依赖注入,在此过程过会让大家更了解的视图,最后还会简单的介绍一下自定义的视图辅助器是怎么定义和使用的。
ASP.NET MVC 视图
l 自定义视图引擎
l Razor视图引擎执行过程
l Razor视图的依赖注入、自定义视图辅助器
l 分段、分部视图的使用
l Razor语法、视图辅助器
Razor视图的依赖注入
首先我们来看一下要定义实现依赖注入的功能接口规范和默认实现,示例代码1-1.
代码1-1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<code>using</code> <code>System.Web.Mvc;</code>
<code>using</code> <code>Ninject;</code>
<code> </code>
<code>namespace</code> <code>MvcApplication.Models</code>
<code>{</code>
<code> </code><code>public</code> <code>interface</code> <code>IStringManage</code>
<code> </code><code>{</code>
<code> </code><code>MvcHtmlStringCombinationString(</code><code>string</code> <code>strPar1, </code><code>string</code> <code>strPar2);</code>
<code> </code><code>}</code>
<code> </code><code>public</code> <code>class</code> <code>DefaultStringManage: IStringManage</code>
<code> </code><code>public</code> <code>MvcHtmlString CombinationString(</code><code>string</code> <code>strPar1, stringstrPar2)</code>
<code> </code><code>{</code>
<code> </code><code>returnnew MvcHtmlString(strPar1+ strPar2);</code>
<code> </code><code>}</code>
<code>}</code>
在IStringManage类型中定义了CombinationString()方法,用于将两个字符串类型的数值拼接起来,DefaultStringManage类型就是默认实现了,这里就不多说了。
下面我们再来定义在编译时刻视图将要实现继承的类型,示例代码1-2
代码1-2
<code> </code><code>public</code> <code>abstract</code> <code>class</code> <code>StringManageView : WebViewPage</code>
<code> </code><code>[Inject]</code>
<code> </code><code>public</code> <code>IStringManage StringManage { </code><code>get</code><code>; </code><code>set</code><code>; }</code>
这样的定义起初是不会有什么问题的,因为cshtml视图文件在编译时是继承自WebViewPage类型的,现在我们要让cshtml视图所继承的类型是StringManageView,所以必须让StringmanageView继承自WebViewPage,因为WebViewPage是抽象类型,而我们又不想实现什么所以要定义为抽象类型,在StringManageView类型中,我定义了IStringManage类型的属性StringMange,并且使用IoC框架中的Inject特性来描述它,使的在编译是可以通过IoC来实现属性的依赖注入。
下面我们来看一下视图代码,这里的视图代码还是引用前面篇幅使用最多的一个示例,代码1-3和代码1-4
代码1-3
<code>public</code> <code>ActionResultIndex(List<Product>model)</code>
<code> </code><code>ViewBag.StrPar1 = </code><code>"Thisis"</code><code>;</code>
<code> </code><code>ViewBag.StrPar2 = </code><code>"ViewIoCCase"</code><code>;</code>
<code> </code><code>returnView(model); </code>
代码1-4
<code>@inherits MvcApplication.Models.StringManageView</code>
<code>@{</code>
<code> </code><code>ViewBag.Title = "Index";</code>
<code><</code><code>h2</code><code>></code>
<code> </code><code>Index</</code><code>h2</code><code>></code>
<code>@foreach (varitem in Model)</code>
<code> </code><code><</code><code>h3</code><code>>ID: @item.ID Name:@item.Name</</code><code>h3</code><code>></code>
<code><</code><code>h2</code><code>>@StringManage.CombinationString(ViewBag.StrPar1,ViewBag.StrPar2)</</code><code>h2</code><code>></code>
控制器方法部分的代码定义是没有问题的,在代码1-4,也就是Index视图的定义中通过@inherits指令来使视图文件在编译时继承自某个类型,以及在下面的使用中用到了StringManage属性,并且还调用了方法,这里看起来都没什么问题,但是放在这里用就有问题了,因为上面使用了foreach来遍历Model,在我们定义StringManageView的时候并没有对Model的类型做约束什么的,而控制器方法中也是需要将List<Product>类型传递到视图的,这里就引起了冲突,图1.
图1
<a href="http://s3.51cto.com/wyfs02/M02/40/3E/wKiom1POTfTjiK1vAAFnLrzmTGY506.jpg" target="_blank"></a>
遇到这种情况我们只需修改一下代码1-2中的定义,让Model类型是在编译时是可确定的而不是object类型,来看代码1-5
代码1-5
<code>public</code> <code>abstract</code> <code>class</code> <code>StringManageView: WebViewPage<dynamic></code>
<code> </code><code>[Inject]</code>
<code> </code><code>public</code> <code>IStringManage StringManage { </code><code>get</code><code>; </code><code>set</code><code>; }</code>
对的,让StringManageView实现泛型的WebViewPage就可以了,上个篇幅中视图的基类也都是这样定义的,不是说上面代码1-2定义的就是错误的,定义的没有错,只是应用的视图不合适,因为Index视图本身需要对Model做一些操作,又不想因为为视图添加的依赖注入功能而修改所以才会引起的这个错误,如果放在普通的视图里使用那是没有问题的。
上面这些都定义好了,下面我们需要实现自定义的IDependencyResolver类型,目的在于绑定我们需要进行依赖注入的功能模块到IoC中,代码1-6.
代码1-6
18
19
20
21
22
23
24
25
26
27
28
29
30
<code>namespace</code> <code>MvcApplication.CustomDependencyResolver</code>
<code> </code><code>public</code> <code>class</code> <code>NinjectDependencyResolver:IDependencyResolver</code>
<code> </code><code>privateIKernel Kernel;</code>
<code> </code><code>publicNinjectDependencyResolver()</code>
<code> </code><code>Kernel = newStandardKernel();</code>
<code> </code><code>AddBinding();</code>
<code> </code><code>privatevoid AddBinding()</code>
<code> </code><code>Kernel.Bind<Models.IStringManage>().To<Models.DefaultStringManage>();</code>
<code> </code><code>public</code> <code>object</code> <code>GetService(TypeserviceType)</code>
<code> </code><code>returnthis.Kernel.TryGet(serviceType);</code>
<code> </code><code>public</code> <code>IEnumerable<</code><code>object</code><code>>GetServices(Type serviceType)</code>
<code> </code><code>returnthis.Kernel.GetAll(serviceType);</code>
对于代码1-6就不作过多的解释了,在控制器激活部分都讲解过近乎类似的注入封装类型。
最后我们在Global.asax文件的Application_Start()方法中,将Model绑定器和NinjectDependencyResolver类型添加中MVC框架中,代码1-7
代码1-7
<code>ModelBinders.Binders.Add(</code><code>typeof</code><code>(List<Product>),</code><code>new</code> <code>CustomListModelBinder());</code>
<code>DependencyResolver.SetResolver(newCustomDependencyResolver.NinjectDependencyResolver());</code>
最后看下结果,图2.
图2
<a href="http://s3.51cto.com/wyfs02/M00/40/3E/wKioL1POTzagPr9mAADcTdSZOnk179.jpg" target="_blank"></a>
自定义视图辅助器
实际上自定义视图辅助器就是扩展方法的定义,首先我们来看定义,实现的功能同代码1-1相同,代码2-1
<code>namespace</code> <code>MvcApplication.CustomHtmlHelper</code>
<code> </code><code>public</code> <code>static</code> <code>class</code> <code>MyCustomHtmlHelper</code>
<code> </code><code>public</code> <code>static</code> <code>MvcHtmlStringCombinationString(</code><code>this</code> <code>HtmlHelperhtmlHelper, </code><code>string</code> <code>strPar1, </code><code>string</code> <code>strPar2)</code>
代码2-1这样的一个类型也就是自定义视图辅助器了,当然了这只是一个简单的示例,现在我们需要把它在视图中使用起来,我们得先把这个自定义的视图辅助器所在的命名空间添加到Views文件中的Web.Config中,代码2-2.
代码2-2
<code> </code><code><</code><code>system.web.webPages.razor</code><code>></code>
<code> </code><code><</code><code>hostfactoryType</code><code>=</code><code>"System.Web.Mvc.MvcWebRazorHostFactory,System.Web.Mvc, Version=3.0.0.0, Culture=neutral,PublicKeyToken=31BF3856AD364E35"</code> <code>/></code>
<code> </code><code><</code><code>pagespageBaseType</code><code>=</code><code>"System.Web.Mvc.WebViewPage"</code><code>></code>
<code> </code><code><</code><code>namespaces</code><code>></code>
<code> </code><code><</code><code>addnamespace</code><code>=</code><code>"System.Web.Mvc"</code> <code>/></code>
<code> </code><code><</code><code>addnamespace</code><code>=</code><code>"System.Web.Mvc.Ajax"</code> <code>/></code>
<code> </code><code><</code><code>addnamespace</code><code>=</code><code>"System.Web.Mvc.Html"</code> <code>/></code>
<code> </code><code><</code><code>addnamespace</code><code>=</code><code>"System.Web.Routing"</code> <code>/></code>
<code> </code><code><</code><code>addnamespace</code><code>=</code><code>"MvcApplication.CustomHtmlHelper"</code><code>/></code>
<code> </code><code></</code><code>namespaces</code><code>></code>
<code> </code><code></</code><code>pages</code><code>></code>
<code> </code><code></</code><code>system.web.webPages.razor</code><code>></code>
然后是在视图中引用扩展方法所处的命名空间,这样配置过后就可以在视图用运用了我们刚刚自定义的视图辅助器了,代码2-3.
代码2-3
<code>@usingMvcApplication.CustomHtmlHelper</code>
<code> </code><code>ViewBag.Title = </code><code>"Index"</code><code>;</code>
<code><h2></code>
<code> </code><code>Index</h2></code>
<code>@</code><code>foreach</code> <code>(varitem </code><code>in</code> <code>Model)</code>
<code> </code><code><h3>ID: @item.ID Name:@item.Name</h3></code>
<code><h2>@StringManage.CombinationString(ViewBag.StrPar1,ViewBag.StrPar2)</h2></code>
<code>@Html.CombinationString(</code><code>"This is a "</code><code>,</code><code>"Case"</code><code>)</code>
最后我们看一下结果如图3.
图3
<a href="http://s3.51cto.com/wyfs02/M01/40/3F/wKiom1POTkuzE0MoAADpgwWbUWU099.jpg" target="_blank"></a>
本文转自jinyuan0829 51CTO博客,原文链接:http://blog.51cto.com/jinyuan/1441508,如需转载请自行联系原作者