Asp.net 5的依赖注入注入系列可以参考链接: [Asp.net 5] DependencyInjection项目代码分析-目录

我们在之前讲微软的实现时,对于OpenIEnumerableService与ClosedIEnumerableService抛下没讲,现在我们就将该部分补充完整。

我们回忆ServiceProvider类的构造函数(对外部使用的)中,注册了IEnumerable<>、new OpenIEnumerableService(_table)的关系。

        public ServiceProvider(IEnumerable<ServiceDescriptor> serviceDescriptors)
{
_root = this;
_table = new ServiceTable(serviceDescriptors); _table.Add(typeof(IServiceProvider), new ServiceProviderService());
_table.Add(typeof(IServiceScopeFactory), new ServiceScopeService());
_table.Add(typeof(IEnumerable<>), new OpenIEnumerableService(_table));
}

ServiceProvider构造函数

因为IEnumerable是泛型,所以我们可以推断OpenIEnumerableService类应该实现IGenericService接口。那么如果我们想查找IEnumerable<T>的实现类中间会有怎样的过程呢?

  • [步骤1]首先系统会查找在ServiceTable内部_services(Dictionary<Type, ServiceEntry>类型)是否有注册过IEnumerable<T>类型,如果有注册直接返回实现类中最后一个。下面是直接包含IEnumerable<T>的原因

    • 系统可以显示的注册IEnumerable<T>,比如services.AddTransient<IEnumerable<T>, List<T>>();
    • 系统也可能之前获取过IEnumerable<T>,所以_services中有上次结果的缓存。
  • [步骤2]如果没有找到相应的IEnumerable<T>,系统会继续通过_genericServices(Dictionary<Type, List<IGenericService>>类型)查找IEnumerable<>,与通过_services方式获取不同,通过_genericServices会获取到的是所有注册过IEnumerable<>类型对应的IGenericService列表(List类型,包含顺序)并不是单一一个IGenericService;之后顺序遍历IGenericService列表,将IEnumerable<T>/IGenericService.GetService()的对应关系添加到services中,重复[步骤1]的操作获取。
    • 如果系统没额外注册IEnumerable<>类型,那么_genericServices的列表中只能获取唯一的注册项OpenIEnumerableService,那么相应的操作则在OpenIEnumerableService中进行。
    • 如果系统额外注册IEnumerable<>类型(假设为GenericService1),那么在注册列表中GenericService1一定排在OpenIEnumerableService之后。所以当获取IEnumerable<T>时,OpenIEnumerableService.GetService()与GenericService1.GetService()返回值一定都会添加到_services中,但是GenericService1.GetService()一定在后面,所以IEnumerable<T>的实现类一定是GenericService1.GetService().CreateCallSite().Invoke()的值;换句话说GenericService1会将OpenIEnumerableService覆盖掉。
    • 由于注册IEnumerable<>会覆盖掉OpenIEnumerableService,所以原则上不允许注册IEnumerable<>类型

OpenIEnumerableService与ClosedIEnumerableService

由于OpenIEnumerableService实现IGenericService接口,所以会返回IService类型的对象,该对象是ClosedIEnumerableService类型。ClosedIEnumerableService类型内部实际上返回的是ServiceTable中_services所有T的注册项,之后以IEnumerable<T>类型返回。

    internal class OpenIEnumerableService : IGenericService
{
private readonly ServiceTable _table; public OpenIEnumerableService(ServiceTable table)
{
_table = table;
} public ServiceLifetime Lifetime
{
get { return ServiceLifetime.Transient; }
} public IService GetService(Type closedServiceType)
{
var itemType = closedServiceType.GetTypeInfo().GenericTypeArguments[]; ServiceEntry entry;
return _table.TryGetEntry(itemType, out entry) ?
new ClosedIEnumerableService(itemType, entry) :
null;
}
}

OpenIEnumerableService

    internal class ClosedIEnumerableService : IService
{
private readonly Type _itemType;
private readonly ServiceEntry _serviceEntry; public ClosedIEnumerableService(Type itemType, ServiceEntry entry)
{
_itemType = itemType;
_serviceEntry = entry;
} public IService Next { get; set; } public ServiceLifetime Lifetime
{
get { return ServiceLifetime.Transient; }
} public IServiceCallSite CreateCallSite(ServiceProvider provider, ISet<Type> callSiteChain)
{
var list = new List<IServiceCallSite>();
for (var service = _serviceEntry.First; service != null; service = service.Next)
{
list.Add(provider.GetResolveCallSite(service, callSiteChain));
}
return new CallSite(_itemType, list.ToArray());
} private class CallSite : IServiceCallSite
{
private readonly Type _itemType;
private readonly IServiceCallSite[] _serviceCallSites; public CallSite(Type itemType, IServiceCallSite[] serviceCallSites)
{
_itemType = itemType;
_serviceCallSites = serviceCallSites;
} public object Invoke(ServiceProvider provider)
{
var array = Array.CreateInstance(_itemType, _serviceCallSites.Length);
for (var index = ; index != _serviceCallSites.Length; ++index)
{
array.SetValue(_serviceCallSites[index].Invoke(provider), index);
}
return array;
} public Expression Build(Expression provider)
{
return Expression.NewArrayInit(
_itemType,
_serviceCallSites.Select(callSite =>
Expression.Convert(
callSite.Build(provider),
_itemType)));
}
}
}

ClosedIEnumerableService

[之前我们介绍ServiceEntry时,明确指出是链表结构,而不是单独存放一个值;其应用在这进行了淋漓尽致的表现]

[Asp.net 5] DependencyInjection项目代码分析4-微软的实现(5)(IEnumerable<>补充)的更多相关文章

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

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

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

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

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

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

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

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

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

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

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

    Microsoft.Framework.DependencyInjection.Ninject 该工程内部共包含5个类文件,底层使用Ninject实现依赖注入,工程截图如下: 从文件命名可以看出,Ni ...

  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判断两个日期是否相等

    大家一定遇到过这样的情况,有两个日期对象,然后需要判断他们是否相等. 例如: var date1 = new Date("2013-11-29"); var date2 = new ...

  2. 在.NET中使用反射实现简易插件机制

    本篇是我学习反射的一个应用小场景而做的学习笔记,主要是一个小的总结,并对各个步骤的记录,以便将来回顾. 一.基础框架-敏捷基础版本 这里假定我们要开发一个记事本,选择Windows Form技术开发, ...

  3. 利用gulp解决前后端分离的header/footer引入问题

    在我们进行前后端完全分离的时候,有一个问题一直是挺头疼的,那就是公共header和footer的引入.在传统利用后端渲染的情况下,我们可以把header.footer写成两个单独的模板,然后用后端语言 ...

  4. Webstorm 10 for mac osx 注册机,序列号,kegen

    小菜最近get到mac体验机会,早就耳闻mac非常适合做开发,于是迫不及待的安装各种开发工具,不知不觉,轮到前端开发神器webstorm了,看了一下官网的价格,心拔凉拔凉的. 果断搜索注册机,搜到的结 ...

  5. IOS SWIFT 启动流程学习

    其实和我们java.c一样通过一个main函数作为入口. main封装在了UIApplicationMain里面.所以后者变成启动入口. 他会扫描Info.plist,找到需要加载的入口storybo ...

  6. C语言 · 寻找数组中的最大值

    问题描述 对于给定整数数组a[],寻找其中最大值,并返回下标. 输入格式 整数数组a[],数组元素个数小于1等于100.输出数据分作两行:第一行只有一个数,表示数组元素个数:第二行为数组的各个元素. ...

  7. Android开发学习之路-LruCache使用和源码分析

    LruCache的Lru指的是LeastRecentlyUsed,也就是近期最少使用算法.也就是说,当我们进行缓存的时候,如果缓存满了,会先淘汰使用的最少的缓存对象. 为什么要用LruCache?其实 ...

  8. ArchLinux+Win10双系统的Grub配置

    解决:ArchLinux+Win10双系统,Grub设置 原装的Win10,装完ArchLinux后,要进入Win10一段时间只能通过boot选择. Grub的菜单里并没有.Grub安装过程是参考wi ...

  9. C#字符串排序效率

    前几天看到个node.js和C#比较性能的文章,在那篇文章中C#的性能居然输了,按理说这是不可能的,除非有什么特殊的情况拖慢了性能.查看其异步的写法,最终发现没有什么问题,起码不是主要问题.后来用VS ...

  10. 模拟image的ajaxPrefilter与ajaxTransport处理

    ////////////////////////////////////////////////////////////////// // options 是请求的选项 // // originalO ...