Microsoft.Framework.DependencyInjection.Ninject

该工程内部共包含5个类文件,底层使用Ninject实现依赖注入,工程截图如下:

从文件命名可以看出,NinjectServiceProvider和NinjectServiceScopeFactory分别是接口IServiceProvider和IServiceScopeFactory的实现类。

(IServiceScope接口的实现类作为NinjectServiceScopeFactory内部类而存在,没有作为单独的文件。)

而工程的入口依旧是Registration为结尾的NinjectRegistration类。

NinjectRegistration

文件代码如下

 public static class NinjectRegistration
{
public static void Populate(this IKernel kernel, IEnumerable<ServiceDescriptor> descriptors)
{
kernel.Load(new ServiceProviderNinjectModule(descriptors));
} public static IBindingNamedWithOrOnSyntax<T> InRequestScope<T>(
this IBindingWhenInNamedWithOrOnSyntax<T> binding)
{
return binding.InScope(context => context.Parameters.GetScopeParameter());
} internal static ScopeParameter GetScopeParameter(this IEnumerable<IParameter> parameters)
{
return (ScopeParameter)(parameters
.Where(p => p.Name == typeof(ScopeParameter).FullName)
.SingleOrDefault());
} internal static IEnumerable<IParameter> AddOrReplaceScopeParameter(
this IEnumerable<IParameter> parameters,
ScopeParameter scopeParameter)
{
return parameters
.Where(p => p.Name != typeof(ScopeParameter).FullName)
.Concat(new[] { scopeParameter });
}
}

静态方法:

Populate:作用是将ServiceProviderNinjectModule注册到内核中。对于Ninject大多数时候都会将自定义继承自NinjectModule的类注册到内核中,并且使用该继承类管理依赖注入。

GetScopeParameter:获取所有“ScopeParameter”类型的参数

AddOrReplaceScopeParameter:获取所有“ScopeParameter”参数,并且将它替换成传入的“scopeParameter”。

InRequestScope:在“binding.InScope”中添加“ScopeParameter”。

总体说来,该文件都是些辅助方法,也是对外初始化的接口。

ServiceProviderNinjectModule

文件代码如下:

internal class ServiceProviderNinjectModule : NinjectModule
{
private readonly IEnumerable<ServiceDescriptor> _serviceDescriptors; public ServiceProviderNinjectModule(
IEnumerable<ServiceDescriptor> serviceDescriptors)
{
_serviceDescriptors = serviceDescriptors;
} public override void Load()
{
foreach (var descriptor in _serviceDescriptors)
{
IBindingWhenInNamedWithOrOnSyntax<object> binding; if (descriptor.ImplementationType != null)
{
binding = Bind(descriptor.ServiceType).To(descriptor.ImplementationType);
}
else if (descriptor.ImplementationFactory != null)
{
binding = Bind(descriptor.ServiceType).ToMethod(context =>
{
var serviceProvider = context.Kernel.Get<IServiceProvider>();
return descriptor.ImplementationFactory(serviceProvider);
});
}
else
{
binding = Bind(descriptor.ServiceType).ToConstant(descriptor.ImplementationInstance);
} switch (descriptor.Lifetime)
{
case ServiceLifetime.Singleton:
binding.InSingletonScope();
break;
case ServiceLifetime.Scoped:
binding.InRequestScope();
break;
case ServiceLifetime.Transient:
binding.InTransientScope();
break;
}
} Bind<IServiceProvider>().ToMethod(context =>
{
var resolver = context.Kernel.Get<IResolutionRoot>();
var inheritedParams = context.Parameters.Where(p => p.ShouldInherit); var scopeParam = new ScopeParameter();
inheritedParams = inheritedParams.AddOrReplaceScopeParameter(scopeParam); return new NinjectServiceProvider(resolver, inheritedParams.ToArray());
}).InRequestScope(); Bind<IServiceScopeFactory>().ToMethod(context =>
{
return new NinjectServiceScopeFactory(context);
}).InRequestScope();
}
}

虽然代码量相对多一些,不过内部逻辑也很简单。

首先是按照惯例,将表示依赖注入关系的IEnumerable<ServiceDescriptor>类型参数传入。

之后重载Load方法,在内核调用时根据IEnumerable<ServiceDescriptor>初始化依赖注入关系。

Load方法,首先遍历IEnumerable<ServiceDescriptor> 对于每一个ServiceDescriptor对象,按照实现类类型、实现类工厂、实现类实例的顺序注入,之后设置相应的生命周期。最后注入IServiceProvider为NinjectServiceProvider、IServiceScopeFactory为NinjectServiceScopeFactory类型。

*需要注意的是NinjectServiceProvider和NinjectServiceScopeFactory注入的范围是Scope,并且对于每个NinjectServiceProvider都会替换其scopeParam参数,以保证其Scope范围。

NinjectServiceProvider

internal class NinjectServiceProvider : IServiceProvider
{
private static readonly MethodInfo _getAll; private readonly IResolutionRoot _resolver;
private readonly IParameter[] _inheritedParameters;
private readonly object[] _getAllParameters; static NinjectServiceProvider()
{
_getAll = typeof(ResolutionExtensions).GetMethod(
"GetAll", new Type[] { typeof(IResolutionRoot), typeof(IParameter[]) });
} public NinjectServiceProvider(IResolutionRoot resolver, IParameter[] inheritedParameters)
{
_resolver = resolver;
_inheritedParameters = inheritedParameters;
_getAllParameters = new object[] { resolver, inheritedParameters };
} public object GetService(Type type)
{
return GetSingleService(type) ??
GetLast(GetAll(type)) ??
GetMultiService(type);
} private object GetSingleService(Type type);
private IEnumerable GetMultiService(Type collectionType);
private IEnumerable GetAll(Type type);
private static object GetLast(IEnumerable services);
}

该类首先静态构造函数在类文件加载时,进行初始化,将范性扩展方法“IEnumerable<T> GetAll<T>(this IResolutionRoot root, params IParameter[] parameters)” 赋值给内部变量“_getAll”。之后类构造函数初始化,该构造函数将IResolutionRoot _resolver赋值(依赖注入的内部的核心控制类),IParameter[] _inheritedParameters赋值,并且为“_getAll”的参数初始化_getAllParameters。

当系统获取实现类实例时,系统调用GetService(Type)方法。

  • 系统首先调用GetSingleService[内部为_resolver.TryGet(type, _inheritedParameters)],获取简单类型的实现。
  • 如果简单类型为空,则获取将泛型类型的数据。将GetAll方法的T类型换成实际的类型,之后获取所有该类型的实例,最后去所有实例中最后一个。
private IEnumerable GetAll(Type type)
{
var getAll = _getAll.MakeGenericMethod(type);
return (IEnumerable)getAll.Invoke(null, _getAllParameters);
}
  • 如果获取所有该类型实例仍然为空,则该Type类型可能为IEnumerable<T>类型,则获取T类型,调用GetAll方法。

NinjectServiceScopeFactory和NinjectServiceScope

这里俩个类继承自IServiceScopeFactory和IServiceScope。

NinjectServiceScopeFactory仅仅在内部创建了NinjectServiceScope对象,代码如下:

public NinjectServiceScopeFactory(IContext context)
{
_resolver = context.Kernel.Get<IResolutionRoot>();
_inheritedParameters = context.Parameters.Where(p => p.ShouldInherit);
} public IServiceScope CreateScope()
{
return new NinjectServiceScope(_resolver, _inheritedParameters);
}

而在NinjectServiceScope内部,每次都是使用新的ScopeParameter参数构建NinjectServiceProvider实例,以确保每次都开启一个新的Scope。代码如下:

 public NinjectServiceScope(
IResolutionRoot resolver,
IEnumerable<IParameter> inheritedParameters)
{
_scope = new ScopeParameter();
inheritedParameters = inheritedParameters.AddOrReplaceScopeParameter(_scope);
_serviceProvider = new NinjectServiceProvider(resolver, inheritedParameters.ToArray());
}
*AddOrReplaceScopeParameter当ScopeParameter类型参数存在时,将其替换成_scope,不存在时添加。

[Asp.net 5] DependencyInjection项目代码分析3-Ninject的更多相关文章

  1. [Asp.net 5] DependencyInjection项目代码分析-目录

    微软DI文章系列如下所示: [Asp.net 5] DependencyInjection项目代码分析 [Asp.net 5] DependencyInjection项目代码分析2-Autofac [ ...

  2. [Asp.net 5] DependencyInjection项目代码分析4-微软的实现(5)(IEnumerable<>补充)

    Asp.net 5的依赖注入注入系列可以参考链接: [Asp.net 5] DependencyInjection项目代码分析-目录 我们在之前讲微软的实现时,对于OpenIEnumerableSer ...

  3. [Asp.net 5] DependencyInjection项目代码分析4-微软的实现(3)

    这个系列已经写了5篇,链接地址如下: [Asp.net 5] DependencyInjection项目代码分析 [Asp.net 5] DependencyInjection项目代码分析2-Auto ...

  4. [Asp.net 5] DependencyInjection项目代码分析4-微软的实现(4)

    这个系列已经写了6篇,链接地址如下: [Asp.net 5] DependencyInjection项目代码分析 [Asp.net 5] DependencyInjection项目代码分析2-Auto ...

  5. [Asp.net 5] DependencyInjection项目代码分析4-微软的实现(2)

    在 DependencyInjection项目代码分析4-微软的实现(1)中介绍了“ServiceTable”.“ServiceEntry”.“IGenericService”.“IService”. ...

  6. [Asp.net 5] DependencyInjection项目代码分析

    最近在研究开源代码,正好发现Asp.net5的源码,下载地址:https://github.com/aspnet. 今天主要讲的是DependencyInjection这部分,抛砖引玉,供大家参考,也 ...

  7. [Asp.net 5] DependencyInjection项目代码分析2-Autofac

    Microsoft.Framework.DependencyInjection.Autofac源码分析 该工程只有一个代码静态类AutofacRegistration,但是该类有3个扩展方法,以及3个 ...

  8. [Asp.net 5] DependencyInjection项目代码分析4-微软的实现(1)

    前面俩种实现中,很多内部细节都无法知道,微软的框架也是为了屏蔽具体实现,只让我们关注接口.但是人都是充满好奇的,依赖注入到底是怎么实现的呢? 微软又有怎样的实现呢?下面就为大家一一呈现(说实话,代码真 ...

  9. Jenkins+Gradle+Sonar进行Java项目代码分析

    Jenkins+Maven+Sonar与Jenkins+Gradle+Sonar配置方法很相似,区别就是Java项目所用的编译工具不同,一个是maven,一个是gradle 使用maven编译工具的可 ...

随机推荐

  1. JavaScript思维导图—Window对象

    JavaScript思维导图-来自@王子墨http://julying.com/blog/the-features-of-javascript-language-summary-maps/

  2. Linux cat命令

    200 ? "200px" : this.width)!important;} --> 介绍 cat命令经常会用来查看一个文件的内容,并且结合它本身的一些参数经常可以用来做一 ...

  3. Oracle 权限(grant、revoke)

    200 ? "200px" : this.width)!important;} --> 数据库版本:11GR2 一.介绍 在oracle中没有其他数据库系统中的数据库的概念, ...

  4. [每日电路图] 8、三轴加速度计LIS3DH电路图及功耗等指标

    看TI的官网资料:http://www.st.com/web/en/catalog/sense_power/FM89/SC444/PF250725 一.初次接触关注的信息: 1.1.概述中的关键信息 ...

  5. 如何为编程爱好者设计一款好玩的智能硬件(九)——LCD1602点阵字符型液晶显示模块驱动封装(下)

    六.温湿度传感器DHT11驱动封装(下):如何为编程爱好者设计一款好玩的智能硬件(六)——初尝试·把温湿度给收集了(下)! 七.点阵字符型液晶显示模块LCD1602驱动封装(上):如何为编程爱好者设计 ...

  6. 在cocos2dx和unity3d之间选择

    人生最纠结的事,莫过于有选择………… cocos2dx和unity3d从某种意义上讲,都很不错.但当面对特定需求以及团队情况的时候,总是能分出高下的. 假设,目标游戏是一款类似 刀塔传奇 的游戏 我们 ...

  7. AngularJS入门教程1--配置环境

    首先需要下载AngualrJS,下载地址 https://angularjs.org/ 官方网站提供2种下载使用AngularJS方法: 1. 去GitHub下载 ,点击按钮会跳转到GitHub页面, ...

  8. MySQL主键设计

    [TOC] 在项目过程中遇到一个看似极为基础的问题,但是在深入思考后还是引出了不少问题,觉得有必要把这一学习过程进行记录. MySQL主键设计原则 MySQL主键应当是对用户没有意义的. MySQL主 ...

  9. [HIMCM暑期班]第1课:概述

    作为这个系列的开始,我会把每一节课上过的内容,与同学们互动后发现他们的闪光点记录下来,以后其他要准备该比赛的人借鉴和参考. 第一节课是概述,主要讲什么是数学建模,还有建模可以帮助我们做什么.举了三个例 ...

  10. IOS Animation-动画基础、深入

    1. Model Layer Tree(模型层树)和Presentation Layer Tree(表示层树) CALayer是动画产生的地方.当我们动画添加到Layer时,是不直接修改layer的属 ...