动态代理中的jdk和Cglib的简单实现。

1. 动态代理,原理不多说了,网上理论很多,直接上代码。

动态代理方式,增强类通过实现InvocationHandler来增强目标类方法.
采用实现接口的方式使增强类依旧可以继承其他类或实现其他接口.

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

//  .动态代理模式
public class Count {

    public static void main(String[] args) {
        SayProxy proxy = new SayProxy(new SayImpl());
        // .获取代理对象
        SayImpl newProxyInstance =(SayImpl) Proxy.newProxyInstance(proxy.getClass().getClassLoader(), proxy
                .getClass().getInterfaces(), proxy);
        // .调用方法(此处方法已经被增强过了)
        newProxyInstance.say();
    }
}

interface Say {
    void say();
}

class SayImpl implements Say {
    @Override
    public void say() {
        System.out.println("我是原本的方法体!");
    }
}

// .代理类,实现增强。
class SayProxy implements InvocationHandler {

    private Object object;

    /**
     * 创建一个新的实例 SayProxy,根据传入类不同,实现动态代理。
     */

    public SayProxy(Object object) {
        this.object = object;
    }

    /**
     * 实现增强
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        System.out.println("我已经前置增强完了");
        
        method.invoke(proxy, args);
        System.out.println("我已经后置增强完了");
        return null;
    }

}

2. Cglib的实现:

需要为代理类指定父类.

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 Count {

    public static void main(String[] args) {
        SayImpl sayImpl = new SayImpl();
        SayProxy proxy = new SayProxy(sayImpl);
        Enhancer enhancer = new Enhancer(); // 创建加强器,用来创建动态代理类
        enhancer.setSuperclass(sayImpl.getClass()); // 为加强器指定要代理的业务类(即:为下面生成的代理类指定父类)
        // 设置回调:对于代理类上所有方法的调用,都会调用CallBack,而Callback则需要实现intercept()方法进行拦
        enhancer.setCallback(proxy);
        SayImpl SayImplProxy = (SayImpl)enhancer.create();
        SayImplProxy.say();
    }
}

interface Say {
    void say();
}

class SayImpl implements Say {
    @Override
    public void say() {
        System.out.println("我是原本的方法体!");
    }
}

// .代理类,实现增强。
class SayProxy implements MethodInterceptor {

    private Object object;

    /**
     * 创建一个新的实例 SayProxy,根据传入类不同,实现动态代理。
     */

    public SayProxy(Object object) {
        this.object = object;
    }

    @Override
    public Object intercept(Object arg0, Method arg1, Object[] arg2,
            MethodProxy arg3) throws Throwable {
        System.out.println("前置增强");
        arg3.invokeSuper(arg0, arg2); // 调用业务类(父类中)的方法
        System.out.println("后置增强");
        return null;
    }

}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 前言 在上文《编译与链接过程的思考》评论中暴走大牙提到了静态库和动态库依赖的问题,还在群里提了几个测试样例和测试工...
    落影loyinglin阅读 4,645评论 11 40
  • 今年不比往年,往年这个时候我已经坚持运动了几个月,但是今年由于时间和场地的限制,最近才开始持续地运动。于是,肚子上...
    大脑洞呀大脑洞阅读 210评论 0 1
  • 你说 你有美貌 我有善良 你用美貌换我的善良 为自己拆去一道城墙。 我说 你有美貌 我有善良 我用善良换你的美貌 ...
    _Heran阅读 428评论 0 2
  • 【一】 当年我所读的大学是离我家很远的,成都的一所高校。那是我第一次离开爸妈去那么遥远的地方,没有朋友,没有同学,...
    萌薇阅读 11,755评论 136 515
  • 亲爱的陌生人: 你好! 今天是五一假期的第一天,没有出去玩,选择了待在图书馆。并不是我不想出去,也不是我爱学习,只...
    我是陶然阅读 277评论 0 2