静态代理和动态代理

静态代理:

代码实现:

// 书籍接口
public interface Book {
    void saveBook();
}
// 接口实现类
public class BookService implements Book{
    @Override
    public void saveBook(){
        System.out.println("保存图书.........");
    }
}
// 代理类
public class BookServiceProxy implements Book{

    Book book;

    public BookServiceProxy(Book book) {
        this.book = book;
    }

    @Override
    public void saveBook() {
        System.out.println("before save.......");
        book.saveBook();
        System.out.println("after save.......");
    }

    public static void main(String[] args) {
        Book book = new BookService();
        BookServiceProxy bookServiceProxy = new BookServiceProxy(book);
        bookServiceProxy.saveBook();
    }
}
// 输出结果:
before save.......
保存图书.........
after save.......

对原生对象进行增强。
静态代理优点:
1、易于理解和实现
2、代理类和真实类的关系都是编译期决定的、和动态代理比起来减少了额外的开销。
静态代理缺点:
每一个类都需要单独创建一个代理类。

动态代理:
1、JDK的动态代理:

实现原理:
基于JDK内部的反射机制实现,在创建代理实现类时比CGLib要快,创建代理速度快。
代码实现:

/* Book类和BookService类和静态代理的一样 */
public class MyJDKProxy implements InvocationHandler {

    private Object targetObject;

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

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("MyJDKProxy  before save.......");
        Object result = method.invoke(this.targetObject, args);
        System.out.println("MyJDKProxy  after save.......");
        return result;
    }

    public static void main(String[] args) {
        // 获取自己的代理类对象
        MyJDKProxy myJDKProxy = new MyJDKProxy();
        // 获取BookService的代理对象
        Book proxyInstance = (Book) myJDKProxy.getProxyInstance(new BookService());
        // 调用代理对象的方法
        proxyInstance.saveBook();
    }
}
2、CGLIB的动态代理:

实现原理:
CGLib动态代理是通过字节码技术底层生成一个继承代理类的类来实现,然后重写代理类的方法(如果被代理类被final关键字所修饰,那么会失败,如果代理类中方法被final修饰,该方法无法代理成功),但是运行速度比JDK动态代理要快。
代码实现:

public class MyCGLIBProxy implements MethodInterceptor {

    private Object targetObject;

    public Object getProxyInstance(Object targetObject){
        this.targetObject = targetObject;
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(targetObject.getClass());
        // 回调方法的参数为代理类对象MyCGLIBProxy,最后增强目标类调用的是代理类对象MyCGLIBProxy中的intercept方法
        enhancer.setCallback(this);
        // 增强后的目标类
        Object proxyObject = enhancer.create();

        return proxyObject;
    }

    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {

        System.out.println("MyCGLIBProxy  before save.......");
        Object obj = method.invoke(targetObject, objects);
        System.out.println("MyCGLIBProxy  after  save.......");

        return obj;
    }

    public static void main(String[] args) {
        MyCGLIBProxy myCGLIBProxy = new MyCGLIBProxy();
        /* 获取代理对象 */
        Book proxyInstance = (Book) myCGLIBProxy.getProxyInstance(new BookService());
        proxyInstance.saveBook();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Subject类,定义了RealSubject和Proxy的共用接口,这样就在任何使用RealSubject的地方...
    联旺阅读 238评论 0 0
  • 一、代理模式 二、静态代理 1.定义所谓静态代理,即是代码编译时已经写好了代理类。例如我们上文介绍的例子就是静态代...
    多彩海洋阅读 120评论 1 1
  • 代理模式 ​ 代理模式在 Java 开发中是一种比较常见的设计模式。设计目的旨在为服务类与客户类之间插入其他功...
    dirft_din阅读 255评论 0 0
  • 表情是什么,我认为表情就是表现出来的情绪。表情可以传达很多信息。高兴了当然就笑了,难过就哭了。两者是相互影响密不可...
    Persistenc_6aea阅读 126,054评论 2 7
  • 16宿命:用概率思维提高你的胜算 以前的我是风险厌恶者,不喜欢去冒险,但是人生放弃了冒险,也就放弃了无数的可能。 ...
    yichen大刀阅读 6,106评论 0 4