Java动态代理实现方法小结

 更新时间:2019年02月22日 10:23:54   作者:changuncle  
这篇文章主要介绍了Java动态代理实现方法,结合实例形式总结分析了java基于JDK、CGLIB及CGLIB实现动态代理的相关操作技巧,需要的朋友可以参考下

本文实例讲述了Java动态代理实现方法。分享给大家供大家参考,具体如下:

静态代理了解的差不多了,但是对于动态代理理解的还不是很通透,这里先把一些常用的动态代理实现方法记录下来,日后时常看看争取早日融会贯通。

1、JDK实现动态代理

主要使用了Proxy.newProxyInstance()方法,该方法的官方解释为:返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。

public interface ISomeService {
  String doFirst();
  void doSecond();
  String doThird();
}
//目标类:代理类要增强的类
public class SomeServiceImpl implements ISomeService {
  @Override
  public String doFirst() {
    return "AAAbbb";
  }
  @Override
  public void doSecond() {
    System.out.println("SomeServiceImpl:执行doSecond()");
  }
  @Override
  public String doThird() {
    return "aaa";
  }
}
public class Mytest {
  public static void main(String[] args) {
    ISomeService target = new SomeServiceImpl();
    ISomeService someService = (ISomeService) Proxy.newProxyInstance(
        target.getClass().getClassLoader(),
        target.getClass().getInterfaces(),
        new InvocationHandler() {
          // proxy:代理对象
          // method:目标方法
          // args:目标方法的参数列表
          @Override
          public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            Object result = method.invoke(target, args);
            if(result!=null) {
              result=((String)result).toUpperCase();
            }
            return result;
          }
        });
    System.out.println(someService.doFirst());
    someService.doSecond();
    System.out.println(someService.doThird());
  }
}

2、CGLIB实现动态代理(没接口)

使用JDK的Proxy实现动态代理,要求目标类与代理类实现相同的接口,若目标类不存在接口,则无法使用该方式实现。

对于没有接口的类,要为其创建动态代理,就要使用CGLIB来实现。CGLIB动态代理的生成原理是生成目标类的子类,而子类是增强过的,这个子类对象就是代理对象。使用CGLIB生成代理类,要求目标类必须能被继承,因此不能是final类。

//目标类:代理类要增强的类
public class SomeService {
  public String doFirst() {
    System.out.println("SomeServiceImpl:执行doFirst()");
    return "AAAbbb";
  }
  public void doSecond() {
    System.out.println("SomeServiceImpl:执行doSecond()");
  }
}
//注意:使用Cglib动态代理,要求目标类不能是final的
//Cglib动态代理的增强原理是:子类增强父类,所以目标类必须能被继承
public class CglibFactory implements MethodInterceptor {
  private SomeService target;
  public CglibFactory() { }
  public CglibFactory(SomeService target) {
    this.target = target;
  }
  public SomeService myCglibCreator() {
    Enhancer enhancer = new Enhancer();
    //指定父类,即目标类。因为Cglib动态代理增强的原理是:子类增强父类
    enhancer.setSuperclass(SomeService.class);
    //设置回调接口对象
    enhancer.setCallback(this);
    //create()方法用于创建Cglib动态代理对象
    return (SomeService)enhancer.create();
  }
  //回调函数的执行条件:代理对象执行目标方法时会触发该方法
  @Override
  public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
    Object invoke = method.invoke(target, args);
    if(invoke!=null) {
      invoke=((String)invoke).toUpperCase();
    }
    return invoke;
  }
}
public class Mytest {
  public static void main(String[] args) {
    SomeService target = new SomeService();
    SomeService proxy = new CglibFactory(target).myCglibCreator();
    proxy.doFirst();
  }
}

3、CGLIB实现动态代理(有接口)

在有接口的情况下利用CGLIB实现动态代理跟没有接口的情况下利用CGLIB实现动态代理,其实差不多。

public interface ISomeService {
  String doFirst();
  void doSecond();
  String doThird();
}
//目标类:代理类要增强的类
public class SomeService implements ISomeService {
  public String doFirst() {
    return "AAAbbb";
  }
  public void doSecond() {
    System.out.println("SomeServiceImpl:执行doSecond()");
  }
  @Override
  public String doThird() {
    return "third";
  }
}
//注意:使用Cglib动态代理,要求目标类不能是final的
//Cglib动态代理的增强原理是:子类增强父类,所以目标类必须能被继承
public class CglibFactory implements MethodInterceptor {
  private ISomeService target;
  public CglibFactory() { }
  public CglibFactory(ISomeService target) {
    this.target = target;
  }
  public ISomeService myCglibCreator() {
    Enhancer enhancer = new Enhancer();
    //指定父类,即目标类。因为Cglib动态代理增强的原理是:子类增强父类
    enhancer.setSuperclass(ISomeService.class);
    //设置回调接口对象
    enhancer.setCallback(this);
    //create()方法用于创建Cglib动态代理对象
    return (ISomeService)enhancer.create();
  }
  //回调函数的执行条件:代理对象执行目标方法时会触发该方法
  @Override
  public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
    Object invoke = method.invoke(target, args);
    if(invoke!=null) {
      invoke=((String)invoke).toUpperCase();
    }
    return invoke;
  }
}
public class Mytest {
  public static void main(String[] args) {
    ISomeService target = new SomeService();
    ISomeService proxy = new CglibFactory(target).myCglibCreator();
    System.out.println(proxy.doFirst());
    proxy.doSecond();
    System.out.println(proxy.doThird());
  }
}

更多java相关内容感兴趣的读者可查看本站专题:《Java面向对象程序设计入门与进阶教程》、《Java数据结构与算法教程》、《Java操作DOM节点技巧总结》、《Java文件与目录操作技巧汇总》和《Java缓存操作技巧汇总

希望本文所述对大家java程序设计有所帮助。

相关文章

  • Java中避免写嵌套if样式的代码详解

    Java中避免写嵌套if样式的代码详解

    这篇文章主要给大家介绍了在Java中如何避免写嵌套if样式的相关资料,文中通过示例代码介绍的非常详细,对大家具有一定的参考学习价值,需要的朋友们下面跟着小编一起来学习学习吧。
    2017-07-07
  • 详解Java分布式IP限流和防止恶意IP攻击方案

    详解Java分布式IP限流和防止恶意IP攻击方案

    这篇文章主要介绍了详解Java分布式IP限流和防止恶意IP攻击方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-03-03
  • JavaMail实现发送超文本(html)格式邮件的方法

    JavaMail实现发送超文本(html)格式邮件的方法

    这篇文章主要介绍了JavaMail实现发送超文本(html)格式邮件的方法,实例分析了java发送超文本文件的相关技巧,需要的朋友可以参考下
    2015-05-05
  • SpringBoot JavaMailSender发送邮件功能(实例详解)

    SpringBoot JavaMailSender发送邮件功能(实例详解)

    JavaMailSender是Spring提供的,非常好用的,实现邮件发送的接口 ,这篇文章主要介绍了SpringBoot JavaMailSender发送邮件功能,需要的朋友可以参考下
    2024-03-03
  • 很简单的Java断点续传实现原理

    很简单的Java断点续传实现原理

    这篇文章主要以实例的方式为大家详细介绍了简单的Java断点续传实现原理,感兴趣的小伙伴们可以参考一下
    2016-07-07
  • Java语言求解完美数代码分析

    Java语言求解完美数代码分析

    这篇文章主要介绍了Java语言求解完美数代码分析,具有一定参考价值,需要的朋友可以了解下。
    2017-12-12
  • Java Spring处理循环依赖详解

    Java Spring处理循环依赖详解

    这篇文章主要介绍了Java中的Spring如何处理循环依赖,依赖指的是Bean与Bean之间的依赖关系,关于更多Spring 处理循环依赖的详情,需要的朋友可以参考下面文章具体内容
    2023-04-04
  • Seata AT模式前后镜像是如何生成详解

    Seata AT模式前后镜像是如何生成详解

    这篇文章主要为大家介绍了Seata AT模式前后镜像是如何生成的方法详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-11-11
  • Java对象转Json,关于@JSONField对象字段重命名和顺序问题

    Java对象转Json,关于@JSONField对象字段重命名和顺序问题

    这篇文章主要介绍了Java对象转Json,关于@JSONField对象字段重命名和顺序问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-08-08
  • Java设计模式中的设计原则之合成复用原则详解

    Java设计模式中的设计原则之合成复用原则详解

    这篇文章主要介绍了Java设计模式中的设计原则之合成复用原则详解,原则是尽量使用合成/聚合的方式,而不是使用继承聚合关系表示的是整体和部分的关系,整体与部分可以分开,可以理解为成员变量和当前类的关系就是聚合关系,需要的朋友可以参考下
    2023-11-11

最新评论