JDK动态代理和Cglib动态代理

我们在学习动态代理之前最好要先了解静态代理。如果没有了解过的同学可以看我之前写的静态代理的博客。
传送门:代理模式(Proxy)

已经了解过静态代理的同学我们就可以接着往下面看了。

1.JDK动态代理

jdk动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用业务方法前调用InvocationHandler处理。代理类必须实现InvocationHandler接口,并且,JDK动态代理只能代理实现了接口的类,没有实现接口的类是不能实现JDK动态代理。我们结合下面的示例代码来深入了解一下JDK动态代理。

编写被代理的接口

package com.gw.proxy;

/**
 * @author gw
 * @date 2019/7/8
 */
interface PersonService {
    public String savePerson();

    public void updatePerson();

    public void deletePerson();
}

被代理的实现类

package com.gw.proxy;

/**
 * @author gw
 * @date 2019/7/8
 */
public class PersonServiceImpl implements PersonService {

    @Override
    public String savePerson() {
        System.out.println("添加");
        return "保存成功!";
    }

    @Override
    public void updatePerson() {
        System.out.println("修改");
    }

    @Override
    public void deletePerson() {
        System.out.println("删除");
    }
}

在动态代理在生成代理对象的时候需要一个拦截器 InvocationHandler 因此咱们需要写一个拦截器

package com.gw.proxy;

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

/**
 * @author gw
 * @date 2019/7/8
 */
public class JDKProxy implements InvocationHandler {

    private Object target;

    public Object createProxy(Object target) {
        this.target = target;
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),
                target.getClass().getInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        doSomething();  //一些额外的处理,
        Object returnValue = method.invoke(this.target, args);
        return returnValue;
    }

    private void doSomething() {
        System.out.println("一些额外的逻辑处理什么的");
    }
}

最后测试一下:

package com.gw.proxy;

/**
 * @author gw
 * @date 2019/7/8
 */
public class ProxyTest {
    public static void main(String[] args) {
        new ProxyTest().proxyTest1();
    }

    public void proxyTest1(){
        Object target = new PersonServiceImpl();
        JDKProxy jdkProxy = new JDKProxy();
        PersonService personService = (PersonService)jdkProxy.createProxy(target);
        String returnValue = (String)personService.savePerson();
        System.out.println(returnValue);
    }
}

我们可以看到这个personService就是一个代理对象。实验成功啦。

2.Cglib动态代理

我们同样看例子。被代理的接口和实现类同上就不重复粘贴代码了。只需要修改代理类

package com.gw.proxy;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

/**
 * @author gw
 * @date 2019/7/8
 */
public class CglibProxy implements MethodInterceptor {
    private Object target;

    public CglibProxy(Object target) {
        this.target = target;
    }

    public Object createProxy() {
        Enhancer enhancer = new Enhancer();
        enhancer.setCallback(this);
        enhancer.setSuperclass(this.target.getClass());
        return enhancer.create();
    }

    @Override
    public Object intercept(Object arg0, Method arg1, Object[] arg2,
                            MethodProxy arg3) throws Throwable {
        Object returnValue = arg1.invoke(this.target, arg2);
        return returnValue;
    }
}

测试类:

package com.gw.proxy;

/**
 * @author gw
 * @date 2019/7/8
 */
public class ProxyTest {
    public static void main(String[] args) {
        new ProxyTest().proxyTest2();
    }

    public void proxyTest2(){
        Object target = new PersonServiceImpl();
        CglibProxy cglibProxy = new CglibProxy(target);
        PersonService personService = (PersonService)cglibProxy.createProxy();
        String returnValue = (String)personService.savePerson();
        System.out.println(returnValue);
    }
}

以上就是Jdk动态代理和cglib动态代理的简单实现代码。

3.jdk动态代理和cglib动态代理的区别

  (1)JDK动态代理只能代理实现了接口的类,没有实现接口的类不能实现JDK的动态代理;

  (2)Cglib动态代理是针对类实现代理的,运行时动态生成被代理类的子类拦截父类方法调用,因此不能代理声明为final类型的类和方法

4.Spring在选择用JDK还是CGLiB的依据:

  (1)当Bean实现接口时,Spring就会用JDK的动态代理

  (2)当Bean没有实现接口时,Spring使用CGlib是实现

  (3)可以强制使用CGlib(在spring配置中加入<aop:aspectj-autoproxy proxy-target-class="true"/>)

5.Cglib和JDK谁的效率高?

 (1)使用CGLib实现动态代理,CGLib底层采用ASM字节码生成框架,使用字节码技术生成代理类,在jdk6之前比使用Java反射效率要高。唯一需要注意的是,CGLib不能对声明为final的方法进行代理,因为CGLib原理是动态生成被代理类的子类。
    (2)但是在jdk6、jdk7、jdk8逐步对JDK动态代理优化之后,在调用次数较少的情况下,JDK代理效率高于CGLIB代理效率,只有当进行大量调用的时候,jdk6和jdk7比CGLIB代理效率低一点,但是到jdk8的时候,jdk代理效率高于CGLIB代理,总之,每一次jdk版本升级,jdk代理效率都得到提升,而CGLIB代理消息确有点跟不上步伐。
    (3)jdk创建对象的速度远大于cglib,这是由于cglib创建对象时需要操作字节码。cglib执行速度略大于jdk,所以比较适合单例模式。另外由于CGLIB的大部分类是直接对Java字节码进行操作,这样生成的类会在Java的永久堆中。如果动态代理操作过多,容易造成永久堆满,触发OutOfMemory异常。spring默认使用jdk动态代理,如果类没有接口,则使用cglib。
 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值