zoukankan      html  css  js  c++  java
  • 代理模式之cglib动态代理

    上一篇博客说了实现InvocationHandler接口的jdk动态代理,还有一种实现动态代理的方式则是:通过继承的方式实现的cglib动态代理。


    先在程序中导入cglib的包,cglib-nodep-2.1_3.jar。


    还是和上一个样例一样,差别就在于代理的实现。

    差别:
    JDK的动态代理机制仅仅能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖当中方法实现增强。但由于採用的是继承。所以不能对final修饰的类进行代理。


    測试的代码例如以下:

    package com.yc.advice;
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    import net.sf.cglib.proxy.Enhancer;
    import net.sf.cglib.proxy.MethodInterceptor;
    import net.sf.cglib.proxy.MethodProxy;
    //这个类是一个切面类。完毕的是向目标类的目标方法增加功能(增强)
    public class RightAdvice implements MethodInterceptor{
        //代理模式最核心:代理模式中一定要有目标类的引用
        private Object targetObject; //注意:这个就是目标类的引用
        @Override
        public Object intercept(Object proxy, Method method, Object[] args,
                MethodProxy arg3) throws Throwable {
            String methodName=method.getName();
            //调用目标类的相应的发放那个方法
            Object returnvalue=method.invoke(targetObject, args);
    if(methodName.startsWith("add")||methodName.startsWith("del")||methodName.startsWith("update")){
                //在invoke中增加您要增强的代码
                check();
            }
            return returnvalue;
        }
        //创建一个方法来完毕创建代理对象
        //代理模式最核心:代理模式中一定要有目标类的引用
        public Object createInstance(Object targetObject){
            this.targetObject=targetObject;
            Enhancer enhancer=new Enhancer();
            //targetObject<- productBizProxy 子类
            //TODO 这里要实现推断
        enhancer.setSuperclass(targetObject.getClass());
            enhancer.setCallback(this);
            return enhancer.create();
        }
        public void check(){
            System.out.println("$$$$$$$$$$$$$$$$$$$");
    		System.out.println("权限检查");
    		System.out.println("$$$$$$$$$$$$$$$$$$$");
        }
    }

    事实上代理的核心还是在切面类中要有目标类的引用,这种话就能够操作目标类.

    測试类:

    public class Test {
        public static void main(String[] args) {
            RightAdvice ra=new RightAdvice();
            ProductBizImpl pb=new ProductBizImpl() ;
            ProductBizImpl productBizProxy=(ProductBizImpl) ra.createInstance(pb);
            productBizProxy.addProduct();
            System.out.println("*****************************************88");  
        }
    }

    分析cglib源代码:
    首先是intercept方法,
    这里写图片描写叙述
    通过其凝视的描写叙述,此方法的功能:
    全部生成的代理方法调用此方法取代原有的方法。
    原始的方法能够使用该方法对象的正常反射调用
    public Object intercept(Object obj, java.lang.reflect.Method method, Object[] args,
    MethodProxy proxy) throws Throwable;
    各个參数:
    * @param obj。增强的对象
    * @param Method ,被拦截的方法
    * @param args方法參数数组;
    * @param用来调用超级(非拦截方法)的代理

  • 相关阅读:
    androidstudio gradle下载速度慢
    paddlex 使用-11 实例图像分割
    paddlex 使用-10 语义图像分割
    redis C# Windows下测试环境
    CSS Flex弹性布局(多个div自动换行)
    扩展排序后重新编号
    纯css制作的打勾(√)小图标
    layui表格-template模板的三种用法
    VB中各种数据类型转换函数
    将ACCESS 的数据库中的表的文件 导出了EXCEL格式
  • 原文地址:https://www.cnblogs.com/cxchanpin/p/7063316.html
Copyright © 2011-2022 走看看