Castle最早在2003年诞生于Apache Avalon项目,目的是为了创建一个IOC(控制反转)框架。发展到现在已经有四个组件了,分别是ActiveRecord(ORM组件),Windsor(IOC组件),DynamicProxy(动态代理组件),MonoRail(Web MVC组件)。

>> IOC-Castle Windor学习

一直想研究公司架构,不知从何下手,那就从Global开始吧,首先讲到的是Castle 注入,我在Word里面总结了下,就直接Copy过来了,代码测试正常,有看不懂的欢迎提问~

1.首先在Global里面进行注册:

 private IWindsorContainer _container;

//初始化一个IOC容器

  _container= new WindsorContainer().Install(FromAssembly.This());

//完成IWindsorInstaller接口中的注册

ControllerBuilder.Current.SetControllerFactory(new WindsorControllerFactory(_container.Kernel));

2.需要自定义一个实例:

//在ASP.NET MVC中,每次请求,DefaultControllerFactory都会为我们创建controller实例,我们需要自定义一个派生自DefaultControllerFactory的类,让Castle Windsor帮我们生成controller实例。

using Castle.MicroKernel;

using System;

using System.Collections.Generic;

using System.Linq;

using System.Web;

using System.Web.Mvc;

using System.Web.Routing;

namespace Demo

{

    public class WindsorControllerFactory : DefaultControllerFactory

    {

        private readonly IKernel _kernel;

        public WindsorControllerFactory(IKernel kernel)

        {

            _kernel = kernel;

        }

        protected override IController GetControllerInstance(System.Web.Routing.RequestContext requestContext, System.Type controllerType)

        {

            if (controllerType == null)

            {

                throw new HttpException(, string.Format("当前对{0}的请求不存在", requestContext.HttpContext.Request.Path));

            }

            return (IController)_kernel.Resolve(controllerType);

        }

        public override void ReleaseController(IController controller)

        {

            _kernel.ReleaseComponent(controller);

            base.ReleaseController(controller);

        }

    }

}

3.声明一个类,定义依赖关系

public class TargetInstall : IWindsorInstaller

 {

 public void Install(IWindsorContainer container, IConfigurationStore store)

 {

 container.Register(Classes.FromThisAssembly() //在哪里找寻接口或类

               .BasedOn<IController>() //实现IController接口

               .If(Component.IsInSameNamespaceAs<HomeController>()) //与HomeController在同一个命名空间

               .If(t => t.Name.EndsWith("Controller")) //以"Controller"结尾

               .Configure(c => c.LifestylePerWebRequest()));//每次请求创建一个Controller实例

  container.Register(Classes.FromThisAssembly()

                .Where(t=>t.Name.EndsWith("son"))

                .WithServiceDefaultInterfaces()

                .Configure(c=>c.LifestyleTransient())

                );

 }

 }

上面这个定义的依赖关系仅仅指定了以son结尾的Person类,如果在定义一个其他的不是以son结尾的类,那么就无法实现映射了,有局限性。 接下来修改为另一种方法做映射,这样只要按照指定规则在接口上面做标记就可以实现动态映射了。代码如下:

下面这段代码可以替换掉类TargetInstall 里面的以son结尾做映射的那段代码了:

var componentList = new List<IRegistration>();

            var classes = Assembly.GetExecutingAssembly().GetTypes().Where(p => p.IsVisible).Where(p => ((ServiceAttribute)p.GetCustomAttributes(typeof(ServiceAttribute), false).FirstOrDefault()) != null).ToList();

            foreach (var item in classes)

            {

                var name = item.FullName;

                var baseType = ((ServiceAttribute)item.GetCustomAttributes(typeof(ServiceAttribute), false).First()).BaseType;

                componentList.Add(Component.For(baseType).ImplementedBy(item).Named(name));

            }

            container.Register(componentList.ToArray());

上面那段代码涉及到一个自定义类ServiceAttribute (这个类就是自定义一个属性)

using System;

using System.Collections.Generic;

using System.Linq;

using System.Web;

namespace Demo

{

    public class ServiceAttribute:Attribute

    {

        /// <summary>

        /// 设置或取得基类

        /// </summary>

        public Type BaseType { get; set; }

    }

}

最后在想要实现映射的类里面加标记就行了,比如Person类:

using System;

using System.Collections.Generic;

using System.Linq;

using System.Web;

namespace Demo

{

    [Service(BaseType =typeof(IPerson))]

    public class Person : IPerson

    {

        public string Speak(string name)

        {

            return name;

        }

    }

}

上面是使用代码直接注入,还可以通过配置文件注入,如下:

1.在Global里面进行注册:

 //Castle配置文件注入
_container = new WindsorContainer(ConfigurationManager.AppSettings["IWindsorContainer"]);
_container.Install(FromAssembly.This());
ControllerBuilder.Current.SetControllerFactory(new WindsorControllerFactory(_container.Kernel));//设置自定义控制工厂

2.进行Web配置:

 <add key="IWindsorContainer" value="config\castle.xml" />

根据配置编写如下文件:

>>Castle.xml

<?xml version="1.0" encoding="utf-8" ?>
<castle>
<include uri="file://castle/components.config" />
</castle>

>>Components.config

<?xml version="1.0" encoding="utf-8"?>
<configuration>
<components>
<!--这里就写对应的接口和类,注意id别重复就行了-->
<component id="GoldCRM.Domain.HXDataImport"
service="Demo.IPerson,Demo"
type="Demo.Person,Demo" >
</component>
</components>
</configuration>

3.定义依赖关系:

public class ControllersInstaller : IWindsorInstaller
{
public void Install(IWindsorContainer container, IConfigurationStore store)
{
//实现IController接口
container.Register(Classes.FromThisAssembly()
.BasedOn<IController>()
.LifestyleTransient());
}
}   

4.需要自定义一个实例重写DefaultContgrollerFactory,参考上面即可:

--记得使用配置文件进行注册,每新增一个接口,要记得手动添加到配置文件哦~ 访问时就构造函数访问即可:

private IPerson person;

public HomeController(IPerson person)

{

this.person=person;

}  

好了,Castle注入讲完了~

>>上面主要讲了Castle注入,既然说到这了,就接着上面把【拦截器】讲了吧:

拦截器可以在一个方法执行之前执行某些操作,并可以阻断方法的执行。

1.首先定义一个拦截器类:

 public class ServiceInterceptor : IInterceptor
{
/// <summary>
/// 拦截方法
/// </summary>
/// <param name="invocation"></param>
public void Intercept(IInvocation invocation)
{ invocation.Proceed();
}
}

2.其次在继承IWindsorInstaller的类里面进行注册:,我就在TargetInstall里面直接写了(PS:按道理为了清晰,分工明确是应该单独建立一个类写的):

 public class TargetInstall : IWindsorInstaller
{ private const string Interceptors = "Demo2.ServiceInterceptor";
public void Install(IWindsorContainer container, IConfigurationStore store)
{
//实现IController接口
container.Register(Classes.FromThisAssembly()
.BasedOn<IController>()
.LifestyleTransient()); //注册拦截器
container.Register(Classes.FromThisAssembly()
.BasedOn<IInterceptor>());
//注册Service方法
var componentList = new List<IRegistration>();
var classes = Assembly.GetExecutingAssembly().GetTypes().Where(p => p.IsVisible).Where(p => ((ServiceAttribute)p.GetCustomAttributes(typeof(ServiceAttribute), false).FirstOrDefault()) != null).ToList(); foreach (var item in classes) {
var name = item.FullName;
var baseType = ((ServiceAttribute)item.GetCustomAttributes(typeof(ServiceAttribute), false).First()).BaseType; componentList.Add(Component.For(baseType).ImplementedBy(item).Named(name).Interceptors(Interceptors));
//Interceptors(Interceptors)这段是将拦截器注入
} container.Register(componentList.ToArray());
}
}

Action拦截器参考:http://blog.csdn.net/gulijiang2008/article/details/7427223  

好啦,拦截器就这么多啦,你运行的话,会发现先走拦截器再走方法哒~

接下来讲NHibernate学习→_→

IOC-Castle Windsor映射的更多相关文章

  1. IoC - Castle Windsor 2.1

    找过一些Windsor教程的文章,博客园上TerryLee有写了不少,以及codeproject等也有一些例子,但都讲的不太明了.今天看到Alex Henderson写的一个系列,非常简单明了.下面是 ...

  2. 小白初学Ioc、DI、Castle Windsor依赖注入,大神勿入(不适)

    过了几天,我又来了.上一篇中有博友提到要分享下属于我们abp初学者的历程,今天抽出点时间写写吧.起初,我是直接去看阳光铭睿的博客,看了一遍下来,感觉好多东西没接触过,接着我又去下了github 里面下 ...

  3. 多个IoC容器适配器设计及性能测试(Castle.Windsor Autofac Spring.Core)

    [转]多个IoC容器适配器设计及性能测试和容器选择 1. 采用的IoC容器和版本 Autofac.2.6.3.862 Castle.Windsor.3.1.0 Spring.Core.2.0.0 2. ...

  4. Castle.Windsor IOC/AOP的使用

    Castle最早在2003年诞生于Apache Avalon项目,目的是为了创建一个IOC(控制反转)框架.发展到现在已经有4个组件了,分别是ActiveRecord(ORM组件).Windsor(I ...

  5. Castle Windsor Ioc 一个接口多个实现解决方案

    介绍 Castle Windsor 是微软的Ioc类库,本文主要介绍解决一个接口多个实现的解决方案 接口和类 以下内容不是真实的实际场景,仅仅是提供解决一个接口多个实现的思路. 业务场景类 先假设有一 ...

  6. 避免Castle Windsor引起的内存泄露

    原文地址: http://nexussharp.wordpress.com/2012/04/21/castle-windsor-avoid-memory-leaks-by-learning-the-u ...

  7. Castle Windsor常用介绍以及其在ABP项目的应用介绍

    最近在研究ABP项目,有关ABP的介绍请看阳光铭睿 博客,ABP的DI和AOP框架用的是Castle Windsor下面就对Castle Windsor项目常用方法介绍和关于ABP的使用总结 1.下载 ...

  8. Aspect Oriented Programming using Interceptors within Castle Windsor and ABP Framework AOP

    http://www.codeproject.com/Articles/1080517/Aspect-Oriented-Programming-using-Interceptors-wit Downl ...

  9. 说说ABP项目中的AutoMapper,Castle Windsor(痛并快乐着)

    这篇博客要说的东西跟ABP,AutoMapper和Castle Windsor都有关系,而且也是我在项目中遇到的问题,最终解决了,现在的感受就是“痛并快乐着”. 首先,这篇博客不是讲什么新的知识点,而 ...

随机推荐

  1. Android 从imageview中获得bitmap的方法

    第一种: 使用setDrawingCacheEnabled()和getDrawingCache()这两种方法,第一个是为了设置是否开启缓存,第二个就可以直接获得imageview中的缓存,一般来说需要 ...

  2. spring的HandlerMapping

    handerlMapping意思是处理器映射,是把请求的url地址与方法进行映射,如SimpleUrlHandlerMapping.

  3. 一个不错的学习android的网站

    http://androiddoc.qiniudn.com/guide/topics/ui/overview.html,最近想学下android的开发,找了一下网上的资料,中文的说的觉得太概括,看不太 ...

  4. Java之关于面向对象

    面向对象,呃,别给我说程序员找不到对象,那是windows才会出现的情况~~~ 就简单记下笔记什么的吧. 1.关于定义和赋值 之前总是搞混淆,说到底是没有搞清楚. shit bigOne=new sh ...

  5. Fear No More歌词

      "Fear No More"   Every anxious thought that steals my breath It's a heavy weight upon my ...

  6. USACO 2008 Mar Silver 3.River Crossing 动态规划水题

    Code: #include<cstring> #include<algorithm> #include<cstdio> using namespace std; ...

  7. node——四种注册路由方式

      app.get和app.post 1.请求的方法必须是get/post2.请求的路径的pathname必须等于(====)路径 app.use 1.在进行路由匹配的时候不限定方法,什么请求方法都可 ...

  8. [总结-动态规划]经典DP状态设定和转移方程

    马上区域赛,发现DP太弱,赶紧复习补上. #普通DP CodeForces-546D Soldier and Number Game 筛法+动态规划 待补 UVALive-8078 Bracket S ...

  9. 【转】H5 input search 提交事件

    欲实现一个文字搜索的功能,要求输入时,键盘回车按钮提示显示为“搜索”.效果如下: 开始~ input type=text并不能达到这种效果,google了一下,HTML5 增加的type=search ...

  10. Win32 编程消息常量(C#)

    public class WinMessages { #region 基本消息 public const int WM_NULL = 0x0000; public const int WM_CREAT ...