Spring aop 深入jdk动态代理(自己写动态代理)

JDK动态代理原理

实际上jdk的动态代理很简单,最重要的方法就是ProxyGenerator.generateProxyClass(),生成代理类字节码文件,动态编译之后交给类加载器加载也就是调用defineClass0(),然后实例化就完成了。
http://blog.csdn.net/qq_25235807/article/details/72084759
这是原来写的一个动态代理的过程。接下来主要是仿照jdk动态代理自己实现一下。

第一步:自己的代理类MyProxy

package myproxy;

import java.lang.reflect.Constructor;

public class MyProxy {

    private final static Class[] constructorParams = { MyInvocationHandler.class };
    //我们也将构造私有化
    @SuppressWarnings("unused")
    private MyProxy() {

    }

    protected MyInvocationHandler h;

    protected MyProxy(MyInvocationHandler h) {
        this.h = h;
    }

    public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, MyInvocationHandler h) {
        Object newInstance = null;
        // 获得代理类class对象
        try {
            Class<?> c1 = getProxyClass(loader, interfaces);
            //实例化代理类对象
            Constructor<?> cons = c1.getConstructor(constructorParams);
            newInstance = cons.newInstance(h);
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

        return newInstance;

    }

    private static final Class<?> getProxyClass(ClassLoader loader, Class<?>[] interfaces) throws Exception {
        final String proxyName = "$proxy0";

        Class<?> interfaceClass = null;
        for (Class<?> intf : interfaces) {
            interfaceClass = Class.forName(intf.getName(), false, loader);
            // 判断被代理的类是否为接口
            if (!interfaceClass.isInterface()) {
                throw new Exception(intf + "is not an interface");
            }
        }
        //获得生成的代理的字节码文件路径
        String filePath = MyProxyGenerator.generateProxyClass(proxyName, interfaces);
        //通过类加载器加载字节码文件到内存
        MyClassLoader loader0 = new MyClassLoader(filePath);
        //返回一个代理类的Class对象
        Class<?> proxyClass = loader0.findClass(proxyName);

        return proxyClass;
    }

}

第二步:拼接源文件,并动态的编译生成字节码文件

package myproxy;

import java.io.FileWriter;
import java.io.IOException;
import java.lang.reflect.Method;

import javax.tools.JavaCompiler;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;

public class MyProxyGenerator {
    static final String rn="\r\n";
    
    /**
     * @param proxyName
     * @param interfaces
     * @return 生成代理类字节码文件,实际就是动态拼接字节码文件的过程
     */
    public static String  generateProxyClass(String proxyName, Class<?>[] interfaces) {
        StringBuffer simpleIntfName=new StringBuffer();
        StringBuffer intfName=new StringBuffer();
        for(int i=0;i<interfaces.length;i++){
            intfName.append("import ").append(interfaces[i].getName()).append(";"+rn);
            simpleIntfName.append(interfaces[i].getSimpleName());
            if(i!=interfaces.length-1){
                simpleIntfName.append(",");
            }
        }
        
        String proxyStr="import java.lang.reflect.Method;"
                +rn
                +"import myproxy.MyInvocationHandler;"
                +rn
                +"import myproxy.MyProxy;"
                +rn
                +intfName.toString()
                +rn
                +"public final class "+proxyName +" extends MyProxy implements " +
                simpleIntfName.toString()+" {"
                +rn
                +"public "+proxyName+"(MyInvocationHandler h){"
                +rn
                +" super(h);"
                +rn
                +"}"
                +rn
                +createMethods(interfaces)
                +rn
                +"}";
        String filePath ="F:/EclipseEEWorkPace/DataStru/src/myproxy/";
        
        try {
            FileWriter writer = new FileWriter(filePath+proxyName+".java");
            writer.write(proxyStr);
            writer.flush();
            writer.close();
        } catch (IOException e) {
            
            e.printStackTrace();
        }
    //拿到编译器 
    JavaCompiler javaCompiler = ToolProvider.getSystemJavaCompiler();
    //拿到一个文件管理系统
    StandardJavaFileManager fileMgr = javaCompiler.getStandardFileManager(null, null, null);
    //获得java文件
    Iterable unit = fileMgr.getJavaFileObjects(filePath+proxyName+".java");
    //动态编译
     javaCompiler.getTask(null, 
            fileMgr, 
            null, 
            null, 
            null, 
            unit).call();
        return filePath;
    }

    private static String createMethods(Class<?>[] interfaces) {
        String methodStr="";
        for(Class<?> intf : interfaces ){
            Method[] methods = intf.getMethods();
            for(Method method : methods){
                
                String returnType = method.getReturnType().getSimpleName();
                StringBuffer isreturn =new StringBuffer();
                StringBuffer changeReurn =new StringBuffer();
                if(!returnType.equals("void")){
                    isreturn.append("return ");
                    changeReurn.append("("+returnType+")");
                }
                Class<?>[] parameterTypes = method.getParameterTypes();
                StringBuffer parameterType=new StringBuffer();
                StringBuffer parameterType1=new StringBuffer();
                StringBuffer parameterName =new StringBuffer();
                for(int i=0;i<parameterTypes.length;i++){
                    parameterType1.append(parameterTypes[i].getSimpleName()).append(".class");
                    parameterType.append(parameterTypes[i].getSimpleName())
                    .append(" arg"+i);
                    parameterName.append("arg"+i);
                    if(i!=parameterTypes.length-1){
                        parameterType.append(",");
                        parameterType1.append(",");
                        parameterName.append(",");
                    }
                    
                    
                }
                 methodStr +="public final " +returnType+" "+ method.getName()+"("+parameterType.toString()+") throws Exception{"
                +rn
                +"Method md =" +intf.getSimpleName()+".class.getMethod(\""+method.getName()+"\",new Class[]{"+parameterType1.toString()+"});"
                +rn
                +isreturn.toString()+changeReurn.toString()
                +"this.h.invoke(this, md, new Object[] {"+parameterName.toString()+"});"
                +rn
                + "}"
                +rn;
            }
        }
        return methodStr;
    }
    
    
}

查看反编译后的class文件

import java.lang.reflect.Method;
import myproxy.MyInvocationHandler;
import myproxy.MyProxy;
import proxy.DataService;
import proxy.DataService1;

public final class $proxy0
  extends MyProxy
  implements DataService, DataService1
{
  public $proxy0(MyInvocationHandler paramMyInvocationHandler)
  {
    super(paramMyInvocationHandler);
  }
  
  public final void update(String paramString)
    throws Exception
  {
    Method localMethod = DataService.class.getMethod("update", new Class[] { String.class });
    this.h.invoke(this, localMethod, new Object[] { paramString });
  }
  
  public final int save(String paramString, int paramInt)
    throws Exception
  {
    Method localMethod = DataService.class.getMethod("save", new Class[] { String.class, Integer.TYPE });
    return ((Integer)this.h.invoke(this, localMethod, new Object[] { paramString, Integer.valueOf(paramInt) })).intValue();
  }
  
  public final String create(String paramString)
    throws Exception
  {
    Method localMethod = DataService1.class.getMethod("create", new Class[] { String.class });
    return (String)this.h.invoke(this, localMethod, new Object[] { paramString });
  }
}

第三步:通过类加载器,将class文件装载到内存

package myproxy;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

public class MyClassLoader extends ClassLoader {
    private String proxyClassFilePath;
    FileInputStream fis = null;
    ByteArrayOutputStream baos = null;

    public MyClassLoader(String proxyClassFilePath) {
        this.proxyClassFilePath = proxyClassFilePath;
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        byte[] data = loadClassFile(name);

        return defineClass(name, data, 0, data.length);
    }

    private byte[] loadClassFile(String name) {
        File file = new File(proxyClassFilePath + name + ".class");
        if (file.exists()) {
            try {
                fis = new FileInputStream(file);
                baos = new ByteArrayOutputStream();
                byte[] buf = new byte[1024];
                int len = 0;
                while ((len = fis.read(buf)) != -1) {
                    baos.write(buf, 0, len);
                }
            } catch (FileNotFoundException e) {

                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }

        return baos.toByteArray();
    }

}

获得代理类实例,验证是否成功

package proxy;

import java.lang.reflect.Method;

import myproxy.MyInvocationHandler;

public class DataInvocationHandler implements MyInvocationHandler {
    private DataService dataService;

    public DataInvocationHandler(DataService dataService) {
        this.dataService = dataService;
    }

    private void before() {
        System.out.println("通知类 ,业务方法前调用--before");

    }

    private void after() {
        System.out.println("通知类 ,业务方法后调用--after");

    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Exception {
        before();
        method.invoke(dataService, args);
        after();
        return null;

    }

}
package proxy;

import java.io.IOException;

import myproxy.MyProxy;

public class Main {

    public static void main(String[] args) throws IOException {

        DataService d = (DataService) MyProxy.newProxyInstance(Main.class.getClassLoader(),
                new Class<?>[] { DataService.class, DataService1.class },
                new DataInvocationHandler(new DataServiceImpl()));
        try {
            d.update("name");
        } catch (Exception e) {
            e.printStackTrace();
        }
        /*
         * byte[] bs = ProxyGenerator.generateProxyClass("$proxy1", new
         * Class<?>[]{DataService.class,DataService1.class}); FileOutputStream
         * fs=new FileOutputStream("$proxy1.class"); fs.write(bs); }
         */

    }
}

图片.png

总结

实际上jdk动态代理非常简单,其核心的方法就是ProxyGenerator.generatorProxyClass()方法生成字节码文件。在它的内部会遍历它实现接口的方法,并且在内部会调用实现InvocationHandler接口的代理的invoke方法实现代理。这也是为什么代理类为什么必须继承InInvocationHandler接口的原因,最后通过defineClass0()将字节码文件装载到内存。
但是我们自己实现的动态代理要慢很多,可见动态代理实际上还有很多值得研究的地方,其中一点就是缓存!!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,524评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,869评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,813评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,210评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,085评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,117评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,533评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,219评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,487评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,582评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,362评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,218评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,589评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,899评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,176评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,503评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,707评论 2 335

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,062评论 25 707
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,497评论 18 139
  • title: Jdk动态代理原理解析 tags:代理 categories:笔记 date: 2017-06-14...
    行径行阅读 19,177评论 3 36
  • 写给老公 后天就是女神节,赶在这个日子从北京回来,我其实蛮高兴的,尽管你这个时候赶回来是因为银行的房贷批下来了,要...
    艾雨xz阅读 151评论 2 0
  • 二十年前,一穷二白,听得最多的就是XX迎来了发展的春天。 二十年后,各方面条件大幅改善,寒冬却成了大众口头禅。“就...
    云非云阅读 715评论 0 0