最近在做服务的细粒度治理,统一管理所有服务的方法、参数、返回值信息。方便后续的各个模块之间的对接和协作。

目前系统中所有的服务,管理到接口契约粒度,即服务接口声明和服务接口实现。要做服务的细粒度治理:

首先需要将服务的各个方法信息全部反射出来,然后再统一管理。

大致的思路是这样的:

1. 下载所有最新的服务包(dll文件集合)

2. 获取所有的服务定义信息

3. 反射加载出每个服务详细的方法信息(方法名、参数、返回值等)

一、下载所有最新的服务包

系统中几百个服务,几十个服务组,每个服务都对应一个服务包(dll文件集合),通过FTP将所有最新的服务包下载到本地,释放出来

每个服务包一个文件夹

二、获取所有的服务定义信息

这个也很简单,从主数据库中获取最新的服务定义信息即可,放到本地内存的一个集合中

三、反射加载出每个服务详细的方法信息(方法名、参数、返回值等)

循环遍历每个服务,Reflect每个接口信息,那么问题来了?

很多服务之间是有依赖的,所以有服务组的概念,反射加载时,必须指定:

AppDomain.CurrentDomain.AppendPrivatePath(servicePath);

然后,反射加载各个接口和接口实现

  Assembly asm = Assembly.Load(metadata.Invoke.UserDefineAssemble);
Type type = asm.GetType(metadata.Invoke.UserDefineClass, true, true);

这样做,通过发现Assembly加载不起来,因为反射加载时,很多依赖的dll找不到,或者不是最新版本(服务有多版本设计)。

具体反射的dll加载顺序,请大家参考MSDN:

https://docs.microsoft.com/en-us/dotnet/framework/deployment/how-the-runtime-locates-assemblies

如何解决这个问题?其实很简单的一个需要,解析Dll中的 Method Table.

此时,想到了ILSpy,Reflector,这些组件实现了Dll的反射加载,找找看看相关的SDK。

找着找着发现了Mono.Cecil.

http://www.mono-project.com/docs/tools+libraries/libraries/Mono.Cecil/

Cecil is a library written by Jb Evain to generate and inspect programs and libraries in the ECMA CIL format.

With Cecil, you can load existing managed assemblies, browse all the contained types, modify them on the fly and save back to the disk the modified assembly.

直接Nuget引用。

关键的几个dll:

using Mono.Cecil;

同时,为了方便加载dll和处理服务组的dll,将所有服务包的dll递归放到一个文件夹中,同名Dll以最新修改时间为准。
上述文件准备工作就绪后,可以编码了:

 using HSF.Service.Govermance.SPI;
using Mono.Cecil;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace HSF.Service.Govermance.Service
{
/// <summary>
/// 反射加载工具类
/// </summary>
class ReflectionUtils
{
public static List<HSFServiceMethod> GetMethods(HSFServiceMetadataEntity metaData, string path)
{
var result = new List<HSFServiceMethod>();
var file = Path.Combine(path, metaData.ServiceImplAssembly.Replace(" ", ""));
var assembly = Mono.Cecil.AssemblyDefinition.ReadAssembly(file);
TypeDefinition type = assembly.MainModule.GetType(metaData.ServiceImplClass);
var methods = type.Methods;
foreach (var method in methods)
{
var hsfMethod = new HSFServiceMethod();
hsfMethod.ID = Guid.NewGuid().ToString();
hsfMethod.ServiceID = metaData.ServiceID;
hsfMethod.MethodName = method.Name;
hsfMethod.ReturnType = method.ReturnType.FullName;
hsfMethod.Parameters = new List<HSFServiceParameter>();
var methodName = method.Name;
if (method.Parameters != null && method.Parameters.Count > )
{
foreach (var parameter in method.Parameters)
{
HSFServiceParameter para = new HSFServiceParameter()
{
ID = Guid.NewGuid().ToString(),
MethodID = hsfMethod.ID,
ServiceID = metaData.ServiceID,
ParameterName = parameter.Name,
ParameterOrder = parameter.Sequence,
ParameterType = parameter.ParameterType.FullName,
}; hsfMethod.Parameters.Add(para);
}
} result.Add(hsfMethod);
} return result;
}
}
}

所有的服务的方法、参数、返回值信息全部搞定。

周国庆

2017/9/29

巧用Mono.Cecil反射加载类型和方法信息的更多相关文章

  1. 运用Mono.Cecil 反射读取.NET程序集元数据

    CLR自带的反射机智和API可以很轻松的读取.NET程序集信息,但是不能对程序集进行修改.CLR提供的是只读的API,但是开源项目Mono.Cecil不仅仅可以读取.NET程序集的元数据,还可以进行修 ...

  2. Mono.Cecil

    Mono Cecil十分强大,强大到可以静态注入程序集(注入后生成新的程序集)和动态注入程序集(注入后不改变目标程序集,只在运行时改变程序集行为),它甚至可以用来调试PDB MDB调试符号格式文件. ...

  3. C# Asp.net中的AOP框架 Microsoft.CCI, Mono.Cecil, Typemock Open-AOP API, PostSharp -摘自网络 (可以利用反射 Attribute 进行面向切面编程 可以用在记录整个方法的Log方面)

    Both Microsoft.CCI and Mono.Cecil are low-level, and don't validate produced assemblies. It takes lo ...

  4. 使用Mono Cecil 动态获取运行时数据 (Atribute形式 进行注入 用于写Log) [此文报考 xxx is declared in another module and needs to be imported的解决方法]-摘自网络

    目录 一:普通写法 二:注入定义 三:Weave函数 四:参数构造 五:业务编写 六:注入调用 7.  怎么调用别的程序集的方法示例 8. [is declared in another module ...

  5. 基于Mono.Cecil的静态注入

    Aop注入有2种方式:动态注入和静态注入,其中动态注入有很多实现了 动态注入有几种方式: 利用Remoting的ContextBoundObject或MarshalByRefObject. 动态代理( ...

  6. 利用Mono.Cecil动态修改程序集来破解商业组件(仅用于研究学习)

    原文 利用Mono.Cecil动态修改程序集来破解商业组件(仅用于研究学习) Mono.Cecil是一个强大的MSIL的注入工具,利用它可以实现动态创建程序集,也可以实现拦截器横向切入动态方法,甚至还 ...

  7. 教你怎么用Mono Cecil - 动态注入 (注意代码的注释)

    原文 教你怎么用Mono Cecil - 动态注入 (注意代码的注释) 使用 Mono Cecil 进行反编译:using Mono.Cecil; using Mono.Cecil.Cil; //.. ...

  8. Mono.Cecil 初探(一):实现AOP

    序言 本篇文章介绍基于Mono.Cecil实现静态AOP的两种方式:无交互AOP和交互式AOP. 概念介绍 Mono.Cecil:一个可加载并浏览现有程序集并进行动态修改并保存的.NET框架. AOP ...

  9. 使用 Mono.Cecil 辅助 Unity3D 手游进行性能测试

    Unity3D 引擎在  UnityEngine 名字空间下,提供了  Profiler 类(Unity 5.6 开始似乎改变了这个名字空间),用于辅助对项目性能进行测试.以 Android 平台为例 ...

随机推荐

  1. 002-Apache Maven 构建生命周期

    Maven - 构建生命周期 什么是构建生命周期 构建生命周期是一组阶段的序列(sequence of phases),每个阶段定义了目标被执行的顺序.这里的阶段是生命周期的一部分. 举例说明,一个典 ...

  2. JAVA通过I/O流复制文件

    JAVA通过I/O流复制文件 本文是对字节流操作,可以多音频视频文件进行操作,亲测有效. 个人感觉这个东西就是靠记的, 没什么好解释的,,,, import java.io.File; import ...

  3. Java 逆变与协变

    最近一直忙于学习模电.数电,搞得头晕脑胀,难得今天晚上挤出一些时间来分析一下Java中的逆变.协变.Java早于C#引入逆变.协变,两者在与C#稍有不同,Java中的逆变.协变引入早于C#,故在形式没 ...

  4. Android事件传递机制详解及最新源码分析——Activity篇

    版权声明:本文出自汪磊的博客,转载请务必注明出处. 在前两篇我们共同探讨了事件传递机制<View篇>与<ViewGroup篇>,我们知道View触摸事件是ViewGroup传递 ...

  5. iOS 环信集成单聊界面,出现消息重复问题

    解决办法很简单,数据重复就是EaseMessageViewController和ChatViewController重复调用了这个吧?//通过会话管理者获取收发消息 [self tableViewDi ...

  6. tkinter第三章(单选和多选)RadioButton CheckButton

    最简单的CheckButton多选类 import tkinter as tk #checkButton的内容,多选 root = tk.Tk() v = tk.IntVar()#装整形变量的 #va ...

  7. MPLS LDP随堂笔记2

    前一天排错 Acl 1 匹配所有ospf的数据包 (目的 ospf建立邻居关系 传递路由条目) 2 放行UDP报文 让LDP邻居能互相收发HELLO包 4 放行TCP报文 让LDP邻居能够建立TCP会 ...

  8. 团队作业8——Beta 阶段冲刺6th day

    一.当天站立式会议 二.每个人的工作 (1)昨天已完成的工作(具体在表格中) 完善订单功能 (2)今天计划完成的工作(具体如下) 完善支付功能 (3) 工作中遇到的困难(在表格中) 成员 昨天已完成的 ...

  9. 结对作业(1)----基于GUI的四则运算

    小伙伴:201421123031 余洋 201421123044  潘志坚  题目要求: 我们在个人作业1中,用各种语言实现了一个命令行的四则运算小程序.进一步,本次要求把这个程序做成GUI(可以是W ...

  10. 团队作业9——测试与发布(Beta版本)(含展示博客)

    团队作业9--测试与发布(Beta版) http://www.cnblogs.com/newteam6/p/6938504.html 团队作业9--展示博客 http://www.cnblogs.co ...