自建简单的依赖注入容器Ioc Container

来源:互联网 发布:淘宝上买车险可靠吗 编辑:程序博客网 时间:2024/06/05 23:42

最近研究了Ioc,DI 以及一些开源的框架,以下根据自己的理解写一个简单的Ioc容器。我们在使用一个依赖注入容器时一般都是实现三步:

  1. 创建一个Ioc容器;
  2. 依赖关系注入,这里需要一种collection对象存储这种注入关系;
  3. 根据依赖注入获取对象的实例;


好了,接下来我们可以按照测试驱动开发的模式完成一个简易版本的注入容器:MinContainer



  • 测试代码:
[csharp] view plaincopyprint?
  1. [TestMethod]  
  2.        public void CanNewContainerAndResolveType()  
  3.        {  
  4.            //  
  5.            // TODO: Add test logic here  
  6.            //  
  7.            IMinContainer container = new MinContainer();  
  8.   
  9.            container.RegisterType(typeof(IDal),typeof(OracleDal));  
  10.   
  11.            //var dal = container.Resolve(typeof(IDal));  
  12.            var dal = container.Resolve<IDal>();  
  13.   
  14.            Assert.IsNotNull(container);  
  15.   
  16.            Assert.IsInstanceOfType(dal, typeof(OracleDal));  
  17.   
  18.        }  


  • 定义接口

创建容器时定义了一个接口类型的容器,在这个接口中需要包含抽象方法的定义,看这个接口

[csharp] view plaincopyprint?
  1. public interface IMinContainer:IDisposable  
  2.  {  
  3.      void RegisterType(Type TFrom, Type Tto);  
  4.        
  5.      object Resolve(Type t);  
  6.   
  7.      TType Resolve<TType>();  
  8.  }  


  • 实现接口

接下来就是实现这个接口,

[csharp] view plaincopyprint?
  1. public   class MinContainer:IMinContainer  
  2.     {  
  3.        private bool disposed;  
  4.   
  5.        private readonly Dictionary<Type, TypeKeys> _regkey =  
  6.            new Dictionary<Type, TypeKeys>();  
  7.   
  8.         #region IMinContainer Members  
  9.   
  10.        public void RegisterType(Type TFrom, Type Tto)   
  11.         {  
  12.   
  13.             TypeKeys tk = new TypeKeys() { minType = TFrom, minInstance = CreateInstance(Tto) };  
  14.            if (!_regkey.ContainsKey(TFrom))  
  15.             _regkey.Add(TFrom, tk);  
  16.   
  17.         }  
  18.   
  19.           
  20.         public object Resolve(Type t)  
  21.         {       
  22.             if(_regkey.ContainsKey(t))  
  23.                 return GetInstance(_regkey[t]);  
  24.   
  25.             throw new Exception("Type not registered" + t);  
  26.         }  
  27.         public TType Resolve<TType>()  
  28.         {  
  29.             return (TType)Resolve(typeof(TType));  
  30.         }...  


这里需要一个字典类型的集合器来保持注册的对象
[csharp] view plaincopyprint?
  1. public class TypeKeys  
  2.   {  
  3.   
  4.       public Type minType { getset; }  
  5.       public object minInstance { getset; }//类型实例  
  6.   
  7.   }  

CreateInstance方法是创建对象的实例
[csharp] view plaincopyprint?
  1. private object CreateInstance(Type t)  
  2.        {  
  3.            var ctor = t.GetConstructors().First();  
  4.            var parameters = ctor.GetParameters().Select(p => Resolve(p.ParameterType)).ToArray();  
  5.            return ctor.Invoke(parameters);  
  6.        }  

在Resolve方法中GetInstance方法实现如下,主要是从字典中获取实例。
[csharp] view plaincopyprint?
  1. private object GetInstance(TypeKeys typekey)  
  2.         {  
  3.             return typekey.minInstance ??   
  4.                 (typekey.minInstance = CreateInstance(typekey.minType));  
  5.         }  


  • 测试
好了,至此简单的原理级别的代码完成,测试。



在控制台中进行测试:




  • 小结:

以上代码只是很简陋的原理实现,需要完善泛型实现,生命周期管理等,不过也很简单,大家可以到codeplex站点下载Unit, Munq等实现框架代码。


原创粉丝点击