动态代理模式

动态代理模式示意图

要实现动态代理分为两个步骤:

  • 代理对象和真实对象建立代理关系
  • 实现代理对象的代理逻辑方法

在Java中有多种动态代理技术, 例如: JDK , CGLIB , Javassist , ASM, 其中最常用的动态代理技术有两种:一种是JDK动态代理, 这是JDK自带的功能,; 另一种是CGLIB,这是第三方提供的一个技术, 目前Spring常用JDK和CGLIB, 而MyBatis使用了Javassist, 无论哪种代理技术, 它们的力量都是相似的.

1. JDK动态代理

 JDK动态代理是java.lang.reflect.* 包提供的方式, 它必须借助一个接口才能实现动态代理

public interface StudentService {
    void sayHello();
}

public class StudentServiceImpl implements StudentService{

    @Override
    public void sayHello() {
        System.out.println("Student sayHello.....");
    }
}

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 * 使用JDK实现代理对象模式
 */
public class JdkProxyExample implements InvocationHandler {

    //真实对象
    private Object target = null;

    /**
     * 建立代理对象和真实对象的代理关系,并返回代理对象
     * @param target 真实对象
     * @return 代理对象
     */
    public Object bind(Object target){
        this.target = target;
     //第1个参数:类加载器, 第二个参数:把生成动态代理对象挂在哪些接口下, 第3个参数:实现方法逻辑的代理类的对象
return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this); } /** * * @param proxy 代理对象 * @param method 当前调用的方法 * @param args 当前方法的参数 * @return 代理结果返回 * @throws Throwable 异常 */ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("进入代理逻辑方法"); System.out.println("在调用真实对象方法之前的一些服务"); Object obj = method.invoke(target, args); System.out.println("在调用真实对象方法之后的一些服务"); return obj; } }

 

测试JDK代理

public class JdkProxyDemo {
    public static void main(String[] args) {
        
        JdkProxyExample example = new JdkProxyExample();
        //获取当前对象的代理对象
        StudentService proxy = (StudentService) example.bind(new StudentServiceImpl());
        proxy.sayHello();
    }
}

 结果:

 

2. CGLIB动态代理

JDK动态代理必须提供接口才能使用, 在一些不能提供接口的环境中, 只能采用其他第三方技术, 比如CGLIB动态代理. 它的优势在不需要提供接口, 只要一个非抽象类就能实现动态代理.

public class StudentServiceImpl {
    public void sayHello(){
        System.out.println("Student sayHello");
    }
}

/**
 * cglib实现动态代理
 */
public class CglibProxyExample implements MethodInterceptor {

    /**
     * 生成CGLIB代理对象
     * @param cls Class类
     * @return Class类的CGLIB代理对象
     */
    public Object getProxy(Class cls){
        //CGLIB enhancer增强类对象
        Enhancer enhancer = new Enhancer();
        //设置增强类型
        enhancer.setSuperclass(cls);
        //定义代理逻辑对象为当前对象,要求当前对俩实现MethodInterceptor
        enhancer.setCallback(this);
        //生成并返回代理对象
        return enhancer.create();
    }

    /**
     * 代理逻辑方法
     * @param proxy 代理对象
     * @param method 方法
     * @param args 方法参数
     * @param methodProxy 方法代理
     * @return 代理逻辑返回
     * @throws Throwable 异常
     */
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {

        System.out.println("调用真实对象前.....");

        //GBLIB反射调用真实对象方法
        Object result = methodProxy.invokeSuper(proxy, args);

        System.out.println("调用真实对象后.....");
        return result;
    }
}

public class CglibDemo {
    public static void main(String[] args) {
        CglibProxyExample example = new CglibProxyExample();
        StudentServiceImpl obj = (StudentServiceImpl) example.getProxy(StudentServiceImpl.class);
        obj.sayHello();
    }
}

 

 CGLib创建的动态代理对象性能比JDK创建的动态代理对象的性能高不少,但是CGLib在创建代理对象时所花费的时间却比JDK多得多,所以对于单例的对象,因为无需频繁创建对象,用CGLib合适,反之,使用JDK方式要更为合适一些。同时,由于CGLib由于是采用动态创建子类的方法,对于final方法,无法进行代理

 

posted @ 2018-09-24 00:30  Aaron2705  阅读(140)  评论(0编辑  收藏  举报