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用来调用超级(非拦截方法)的代理

  • 相关阅读:
    AC自动机
    KMP、扩展KMP、MANACHER
    Docker用户身份登录和管理员权限
    Response.Redirect和Server.Transfer比较--(转)
    SQLServer中char、varchar、nchar、nvarchar的区别--(转)
    MsSQL的字段类型--(转)
    读取UEditor编辑框内容到数据库和上传图片的配置
    以做产品的思想分析男女相处之道
    springBoot创建定时任务
    Runnable和Thread的区别
  • 原文地址:https://www.cnblogs.com/cxchanpin/p/7063316.html
Copyright © 2011-2022 走看看