jdk动态代理和cglib动态代理

前言:动态代理作为代理模式的一种扩展形式,广泛应用于框架(尤其是基于AOP的框架)的设计与开发。


jdk动态代理

下面将代码实例展示

package cn.mode.proxy;

/**
 * 接口类
 * @author zzh
 *
 */
public interface People {
    void say(int i);
}
package cn.mode.proxy;

/**
 * 实现类
 * @author zzh
 *
 */
public class Girl implements People {
    @Override
    public void say(int i) {
        if (i == 1) {
            System.out.println("我在吃饭");
        } else {
            System.out.println("我什么事都没做");
        }
    }

}
package cn.mode.proxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.util.Date;

/**
 * 代理类
 * @author zzh
 *
 */
public class SayHandler implements InvocationHandler {

    private Object obj;

    //构造方法传入代理
    public SayHandler(Object obj) {
        this.obj = obj;
    }

    public SayHandler() {
    }


    @Override
    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        System.out.println("开始代理,时间:" + new Date());
        Object result = method.invoke(obj, args);
        System.out.println("代理结束");
        return result;
    }

}
package cn.mode.proxy;

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

/**
 * 测试类
 * @author zzh
 *
 */
public class Test {

    public static void main(String[] args) {

        People people = new Girl();
        //代理类
        InvocationHandler handler = new SayHandler(people);
        //代理对象
        People proxy = (People) Proxy.newProxyInstance(People.class.getClassLoader(), new Class[]{People.class}, handler);

        proxy.say(1);
    }
}

输出:
开始代理,时间:Sun Mar 27 21:24:58 CST 2016
我在吃饭
代理结束

jdk动态代理只能代理一个或多个接口,如果需要动态代理具体类或抽象类,可以使用cglib动态代理。


cglib动态代理

看一个例子

package cn.mode.proxy.cglib;

public class People {

    public void say() {
        System.out.println("吃吃");
    }
}
package cn.mode.proxy.cglib;

public class Girl extends People {
    @Override
    public void say() {
        System.out.println("我在吃饭");
    }

}
package cn.mode.proxy.cglib;

import java.lang.reflect.Method;

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

public class PeopleProxy implements MethodInterceptor {

    private Enhancer enhancer = new Enhancer();  
    public Object getProxy(Class<?> clazz){  
        //设置需要创建子类的类  
        enhancer.setSuperclass(clazz); 
        //回调
        enhancer.setCallback(this);  
        //通过字节码技术动态创建子类实例  
        return enhancer.create();  
    } 

    @Override
    public Object intercept(Object obj, Method method, Object[] args,
            MethodProxy proxy) throws Throwable {
        System.out.println("代理前");
        Object object = proxy.invokeSuper(obj, args);
        System.out.println("代理后");
        return object;
    }

}
package cn.mode.proxy.cglib;

public class Test {

    public static void main(String[] args) {
        PeopleProxy proxy = new PeopleProxy();
        People people = (People) proxy.getProxy(Girl.class);
        people.say();


    }
}

输出:
代理前
我在吃饭
代理后

对比

  1. jdk的动态代理使用的反射,cglib采用了非常底层的字节码技术,其原理是通过字节码技术为一个类创建子类,并在子类中采用方法拦截的技术拦截所有父类方法的调用。

  2. jdk动态代理创建的时间的比cglib快,cglib的创建后的使用效率比较高,对于创建后使用频率较多的建议使用cglib。

  3. 由于CGLib由于是采用动态创建子类的方法,对于final方法,无法进行代理。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

久梦歌行

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值