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 服务.总体目标是使客户端和文件系统作为服务器以同样的速度来更新.文件的方法,参数和模型紧密集 ...
随机推荐
- Maven——依赖
一.依赖的范围 依赖的范围有几个可选值,我们用得到的是:compile.test.provided 三个. [1]从项目结构角度理解 compile 和 test 的区别 [2]从开发和运行这两个不同 ...
- js实现在表格中删除和添加一行
<!DOCTYPE html><html> <head> <title> new document </title> <meta ht ...
- redis过期key的清理策略
一,有三种不同的删除策略(1),立即清理.在设置键的过期时间时,创建一个回调事件,当过期时间达到时,由时间处理器自动执行键的删除操作. (2),惰性清理.键过期了就过期了,不管.当读/写一个已经过期的 ...
- using JSTL
http://docs.oracle.com/javaee/5/tutorial/doc/bnake.html JSTL(JSP Standard Tag Library)
- MySQL性能调优与架构设计——第7章 MySQL数据库锁定机制
第7章 MySQL数据库锁定机制 前言: 为了保证数据的一致完整性,任何一个数据库都存在锁定机制.锁定机制的优劣直接应想到一个数据库系统的并发处理能力和性能,所以锁定机制的实现也就成为了各种数据库的核 ...
- DataGridView移动上下行
/// <summary> /// 移动行 /// </summary> /// <param name="i_UpFlag">移动标识,tru ...
- HTML5移动Web开发实战 PDF扫描版
<HTML5移动Web开发实战>提供了应对这一挑战的解决方案.通过阅读本书,你将了解如何有效地利用最新的HTML5的那些针对移动网站的功能,横跨多个移动平台.全书共分10章,从移动Web. ...
- 在云主机后台进行python程序运行
nohup python main.py & nohup liunx自带的命令 注意:后面(&)!
- HackThree
创建自定义ViewGroup 一,概要: 使用自定义View 和ViewGroup组织应用程序布局是一个好方法,定制组件的同时允许开发者提供自定义行为和功能,以后,开发者 在需要创建复杂布局 ...
- [转] MySQL索引原理
MySQL索引原理 B+树索引是B+树在数据库中的一种实现,是最常见也是数据库中使用最为频繁的一种索引.B+树中的B代表平衡(balance),而不是二叉(binary),因为B+树是从最早的平衡二叉 ...