深入理解Spring--动态代理(Jdk和Cglib)

一、原理区别:

java动态代理:是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。

cglib动态代理:是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。

1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP 

2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP 

3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换

ii. 如何强制使用CGLIB实现AOP?
 (1)添加CGLIB库,SPRING_HOME/cglib/*.jar
 (2)在spring配置文件中加入

iii. JDK动态代理和CGLIB字节码生成的区别?
 (1)JDK动态代理只能对实现了接口的类生成代理,而不能针对类
 (2)CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法
   因为是继承,所以该类或方法最好不要声明成final 

二.动态代理,Cglib和Jdk代码实现

1.User接口及实现类

public interface UserService {
    void addUser(String username,String pwd);
}


public class UserServiceImpl implements UserService{
    @Override
    public void addUser(String username, String pwd) {
        System.out.println("传参:"+username+","+pwd);
    }
}

2.Jdk代理


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class JdkProxy implements InvocationHandler{

    //创建代理对象
    private Object target;

    //监听
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("JDK动态代理,监听开始!!!!!!");
        System.out.println("方法调用前,可以做什么");
        Object result = method.invoke(target,args);
        System.out.println("方法调用后,可以做什么");
        System.out.println("JDK动态代理,监听结束!!!!!!");
        return result;
    }
    //定义获取代理对象的方法
    private Object getJdkProxy(Object targetObject){
        this.target = targetObject;
//     * @param   loader the class loader to define the proxy class
//     * @param   interfaces the list of interfaces for the proxy class
//     *          to implement
//     * @param   将方法调用分派到的调用处理程序
        return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),
                targetObject.getClass().getInterfaces(),this);
    }

    public static void main(String[] args) {
        //创建代理对象实例
        JdkProxy jdkProxy = new JdkProxy();
        UserService user = (UserService) jdkProxy.getJdkProxy(new UserServiceImpl());
        user.addUser("root","root");
    }
}

3.Cglib代理

import com.shiro.service.UserService;
import com.shiro.service.UserServiceImpl;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

public class CglibProxy implements MethodInterceptor {

    private Object target;

    /**
     *  拦截方法,监听
     * @param o
     * @param method
     * @param objects
     * @param methodProxy
     * @return
     * @throws Throwable
     */
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("Cglib,监听开始");
        Object invoke = method.invoke(target,objects);//方法执行,参数:target 目标对象 arr参数数组
        System.out.println("Cglib,监听结束");
        return invoke;
    }

    //定义获取代理对象的方法
    public Object getProxy(Object objectTarget){
        this.target = objectTarget;
        //创建抽象类生成实例
        Enhancer enhancer = new Enhancer();
        //设置父类,因为Cglib是针对指定的类生成一个子类,所以需要指定父类
        enhancer.setSuperclass(objectTarget.getClass());
        //设置回调
        enhancer.setCallback(this);
        //创建并返回代理对象实例
        return enhancer.create();
    }

    public static void main(String[] args) {
        CglibProxy cglibProxy = new CglibProxy();
        UserService userService = (UserService) cglibProxy.getProxy(new UserServiceImpl());
        userService.addUser("cglib root","12345");
    }
}

 

你可能感兴趣的:(Spring)