Emit生成特定接口的类
参考 动态生成类
http://www w2bc com/Article/44799
http://www.cnblogs.com/yingql/archive/2009/03/24/1420914.html
http://www.cnblogs.com/BangQ/archive/2011/07/19/2110301.html?spm=5176.100239.0.0.kAe2my
http://www.cnblogs.com/yuming1983/p/3701540.html
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Reflection.Emit;
using System.Text;
using System.Threading.Tasks; namespace ConsoleApplication1
{ public interface IGetData
{
void GetData(string name);
} public class GetDataFromDb : IGetData
{
public void GetData(string name)
{
Console.WriteLine( "Hello " + name);
}
} public class DynamicProxyBuilder
{
private static AssemblyBuilder DynamicAssembly;
private static ModuleBuilder ModuleBuilder; private static Dictionary<string, Type> TypeList = new Dictionary<string, Type>();
public static TTargetInterface GetProxyObject<TTargetInterface, TOriginalClass>() where TTargetInterface :class where TOriginalClass : TTargetInterface
{
Type target = typeof(TTargetInterface);
Type baseType = typeof(TOriginalClass);
CheckParams(target, baseType);
Type proxyType = AutoGenerateProxyClass(target, baseType);
var baseInstance = Activator.CreateInstance(baseType); return Activator.CreateInstance(proxyType, baseInstance ) as TTargetInterface;
} private static Type AutoGenerateProxyClass(Type target, Type baseType)
{
var proxyClassName = baseType.FullName + "Proxy";
if (!TypeList.ContainsKey(proxyClassName))
{
var module = GetDynamicModule();
var typeBuilder = module.DefineType(proxyClassName,
System.Reflection.TypeAttributes.Public | System.Reflection.TypeAttributes.Class,
typeof(System.Object), new Type[] { target }); var fieldBuilder = typeBuilder.DefineField("OriginalObj", target, System.Reflection.FieldAttributes.Public);
CreateConstructorFunctionForType(baseType, typeBuilder, fieldBuilder);
foreach (var methodInfo in target.GetMethods())
{
CreateMethodForType(baseType, typeBuilder, methodInfo, fieldBuilder);
} TypeList.Add(proxyClassName, typeBuilder.CreateType());
} return TypeList[proxyClassName];
} private static void CreateConstructorFunctionForType(Type baseType, TypeBuilder typeBuilder, FieldBuilder fieldBuilder)
{
var objType = typeof(object);
ConstructorInfo objCtor = objType.GetConstructor(new Type[]);
ConstructorBuilder cb = typeBuilder.DefineConstructor(MethodAttributes.Public,
CallingConventions.Standard, new Type[] { baseType });
ILGenerator ilGenerator = cb.GetILGenerator();
ilGenerator.Emit(OpCodes.Ldarg_0);
//先构建object的构造函数
ilGenerator.Emit(OpCodes.Call, objCtor);
ilGenerator.Emit(OpCodes.Ldarg_0);
ilGenerator.Emit(OpCodes.Ldarg_1);
//将本类的构建函数的参数赋值给本类的字段
ilGenerator.Emit(OpCodes.Stfld, fieldBuilder);
ilGenerator.Emit(OpCodes.Ret);
} private static void CreateMethodForType(Type baseType, TypeBuilder typeBuilder, MethodInfo method, FieldBuilder fieldBuilder)
{
var parameterInfos = method.GetParameters();
var paramTypes = new Type[parameterInfos.Length];
for (int i = ; i < parameterInfos.Length; i++)
{
paramTypes[i] = parameterInfos[i].ParameterType;
}
//准备好真正要调用的方法
var targetMethod = baseType.GetMethod(method.Name, paramTypes);
//创建代理方法,使用接口中相应方法的信息,并去掉期抽象方法属性
var methodBuilder = typeBuilder.DefineMethod(method.Name, method.Attributes & (~MethodAttributes.Abstract),
method.CallingConvention
, method.ReturnType, paramTypes); var il = methodBuilder.GetILGenerator();
il.EmitWriteLine("I am Proxyer");
//开始向栈中压参数,
//第1个参数 当前是this指针
il.Emit(OpCodes.Ldarg_0);
//压入当前引用的字段值 相当于 this.
il.Emit(OpCodes.Ldfld, fieldBuilder);
for (int i = ; i <= parameterInfos.Length; i++)
{
if (i == )
{ }
else if (i == )
{
il.Emit(OpCodes.Ldarg_1);
}
else if (i == )
{
il.Emit(OpCodes.Ldarg_2);
}
else if (i == )
{
il.Emit(OpCodes.Ldarg_3);
}
else
{
il.Emit(OpCodes.Ldarg_S);
}
} il.Emit(OpCodes.Callvirt, targetMethod);
il.EmitWriteLine("I done it!Bye!!");
il.Emit(OpCodes.Ret); } private static ModuleBuilder GetDynamicModule()
{
if (DynamicProxyBuilder.DynamicAssembly == null)
{
DynamicProxyBuilder.DynamicAssembly = AppDomain.CurrentDomain.DefineDynamicAssembly(new AssemblyName("DynamicAssembly"),
AssemblyBuilderAccess.Run);
DynamicProxyBuilder.ModuleBuilder = DynamicProxyBuilder.DynamicAssembly.DefineDynamicModule("MainModule");
} return DynamicProxyBuilder.ModuleBuilder;
} private static void CheckParams(Type targetType, Type baseType)
{
if (!targetType.IsInterface)
throw new Exception("模板参数必须是接口");
}
} public class Program
{
static void Main(string[] args)
{
var instance = DynamicProxyBuilder.GetProxyObject<IGetData, GetDataFromDb>();
instance.GetData("Aven");
Console.Read();
}
}
}
Emit生成特定接口的类的更多相关文章
- .net aop 操作 切面应用 Castle.Windsor框架 spring 可根据接口 自动生成一个空的实现接口的类
通过unget 安装Castle.Windsor using Castle.DynamicProxy; using System; using System.Collections.Generic; ...
- 使用generatorConfig工具自动生成mybatis的实体类以及dao接口和映射文件
1:数据准备 创建一个数据库表 CREATE TABLE `logininfo` ( `id` ) NOT NULL AUTO_INCREMENT, `username` ) DEFAULT NULL ...
- 2016.12.5 在Eclipse中为实现类impl自动生成对应接口
参考来自:http://jingyan.baidu.com/article/ab69b270d63f572ca6189f51.html 在Spring应用中,常常会用到“接口+实现类”的形式,即要实现 ...
- 5.7 Liquibase:与具体数据库独立的追踪、管理和应用数据库Scheme变化的工具。-mybatis-generator将数据库表反向生成对应的实体类及基于mybatis的mapper接口和xml映射文件(类似代码生成器)
一. liquibase 使用说明 功能概述:通过xml文件规范化维护数据库表结构及初始化数据. 1.配置不同环境下的数据库信息 (1)创建不同环境的数据库. (2)在resource/liquiba ...
- Servlet API遍程常用接口和类
本文主要总结Servlet API遍程常用接口和类 Servlet API http://tomcat.apache.org/tomcat-5.5-doc/servletapi/index.html ...
- (转)beanUtil接口和类(有空的时候去看,到时候删除这个说明)
Jakarta Commons项目提供了相当丰富的API,我们之前了解到的Commons Lang只是众多API的比较核心的一小部分而已.Commons下面还有相当数量的子项目,用于解决各种各样不同方 ...
- BeanUtils接口和类
Jakarta Commons项目提供了相当丰富的API,我们之前了解到的Commons Lang只是众多API的比较核心的一小部分而已.Commons下面还有相当数量的子项目,用于解决各种各样不 ...
- c#实例化继承类,必须对被继承类的程序集做引用 .net core Redis分布式缓存客户端实现逻辑分析及示例demo 数据库笔记之索引和事务 centos 7下安装python 3.6笔记 你大波哥~ C#开源框架(转载) JSON C# Class Generator ---由json字符串生成C#实体类的工具
c#实例化继承类,必须对被继承类的程序集做引用 0x00 问题 类型“Model.NewModel”在未被引用的程序集中定义.必须添加对程序集“Model, Version=1.0.0.0, Cu ...
- Swagger+Spring mvc生成Restful接口文档
简介 Swagger 是一个规范和完整的框架,用于生成.描述.调用和可视化 RESTful 风格的 Web 服务.总体目标是使客户端和文件系统作为服务器以同样的速度来更新.文件的方法,参数和模型紧密集 ...
随机推荐
- 2-配置Andriod环境时的错误。。。Theme.AppCompat.Light
编译或运行时可能会出现错误: Error:Error retrieving parent for item: No resource found that matches the given name ...
- Python pandas DataFrame操作
1. 从字典创建Dataframe >>> import pandas as pd >>> dict1 = {'col1':[1,2,5,7],'col2':['a ...
- Shiro——认证
引入shiro依赖 <!-- shiro --> <dependency> <!-- shiro-core Required in all environments. - ...
- Linux命令累积
常用命令 ipconfig -查看本机ip.接口等信息 ping ip -ping远程服务器或终端 cd ~ -返回根目录 cd .. 返回上级目录 cd ../.. 返回上两级目录 ...
- java并发机制的底层实现原理
volatile是轻量级的synchronized,它在多处理器开发中保证了共享变量的"可见性".可见性是说当一个线程修改一个共享变量时,另外一个线程能读到这个修改的值. vola ...
- 观察者(Observer)模式 * 委托事件
观察者(Observer)模式:定义了一种一对多的依赖关系.让多个观察者对象同时监听某一个主题对象. 这个主题对象发生变化时会通知所有观察者对象,使他们字段更新自己 /* * 抽象主题(Subje ...
- js工具库简单介绍
javascript mvc的解决方案: angularjs, backbone,underscore, 有空的时候了解一下. 移动端的几个需要了解一下,jq mobile, zepto.knocko ...
- 关于使用idea的一些小技巧
1:idea与git同步以后查看修改变化: file --setting--versioncontorller
- C#导出Excel-利用特性自定义数据
网上C#导出Excel的方法有很多.但用来用去感觉不够自动化.于是花了点时间,利用特性做了个比较通用的导出方法.只需要根据实体类,自动导出想要的数据 1.在NuGet上安装Aspose.Cells或 ...
- C# 调接口
上一个项目,需要mvc管理后台调接口项目,以便后期的重构扩展,调研后发现后台用的ajax请求,直接调接口可能会有跨域问题,最终在c#代码中实现了这个需求. 1,Ajax请求后台 将接口所需参数传入 2 ...