美文网首页
JAVA动态代理

JAVA动态代理

作者: LordZhou | 来源:发表于2017-01-17 15:09 被阅读0次

Spring AOP功能最基本的技术要点为动态代理。
当下Java主要有两种动态代理方式

  • 基于接口的JDK动态代理
  • 基于类CGLib动态代理

JDK动态代理

JDK动态代理是在运行时根据类的接口生成新的实现类,让新的实现类对已有对象进行代理。

首先声明两个接口

    public interface Machine {
      public void start();
    }
    public interface Car extends Machine {
      public void drive();
    }

实现类

    public class Bus implements Car {

      @Override
      public void drive() {
        System.out.println("老司机开车啦!");
      }

      @Override
      public void start() {
        System.out.println("开始!!");

      }

    }

代理处理类:

    public class JdkProxy implements InvocationHandler {
      private Object obj;

      @SuppressWarnings( "unchecked" )
      public <T> T getProxy( T obj ) {
        this.obj = obj;
        return (T) Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
      }

      @Override
      public Object invoke( Object proxy, Method method, Object[] args ) throws Throwable {
        System.out.println("动态代理开始!");
        Object reVal = method.invoke(obj, args);
        System.out.println("动态代理结束!");
        return reVal;
      }

    }

Proxy.newProxyInstance是使用当前类的接口生成代理对象。
InvocationHandler 为代理的处理接口。

Main函数

    public static void main( String[] args ) {
        Car car = new Bus();
        JdkProxy jdkProxy = new JdkProxy();
        Car proxyCar = jdkProxy.getProxy(car);
        proxyCar.start();
        proxyCar.drive();
    }
结果

CGLib动态代理

CGLib是生成当前类的子类,让子类对父类进行代理。

引入Jar包

    <dependency>
        <groupId>cglib</groupId>
        <artifactId>cglib</artifactId>
        <version>3.2.4</version>
    </dependency>

被代理的类

    public class Student {
      public void sayHello() {
        System.out.println("hello");
      }
    }

代理处理及代理对象生成

    public class StudentProxy implements MethodInterceptor {
      private Enhancer enhancer;

      public <T> T getProxy( Class<T> clazz ) {
        enhancer = new Enhancer();
        enhancer.setSuperclass(clazz);
        enhancer.setCallback(this);

        return (T) enhancer.create();
      }

      @Override
      public Object intercept( Object object, Method method, Object[] args, MethodProxy proxy ) throws Throwable {
        System.out.println("before");
        proxy.invokeSuper(object, args);
        System.out.println("after");
        return null;
      }

    }

Main函数:

    public static void main( String[] args ) {
        StudentProxy proxy = new StudentProxy();
        Student student = proxy.getProxy(Student.class);
        student.sayHello();
    }

结果:

结果

简单测试两种代理方式的性能:

    /**
     * @author : Zak
     * @date : 2017年1月17日 上午11:52:23
     * @version : 2017年1月17日 Zak 首次创建
     */
    public class Main {

      public static void main( String[] args ) {
        Bus bus = new Bus();
        JdkProxy jdkProxy = new JdkProxy();
        CglibProxy cglibProxy = new CglibProxy();
        long jdkproxyInitTime = 0, cglibproxyinitTime = 0, jdkInvokeTime = 0, cglibInvokeTime = 0, startTime = 0;
        for( int i = 0; i < 1000001; i++ ) {
          System.out.println(i);
          switch(i % 2) {
            case 0:// JDKProxy
              startTime = System.currentTimeMillis();
              Car car = jdkProxy.getProxy(bus);
              jdkproxyInitTime += System.currentTimeMillis() - startTime;

              startTime = System.currentTimeMillis();
              car.drive();
              jdkInvokeTime += System.currentTimeMillis() - startTime;

              break;
            case 1:// CGLib

              startTime = System.currentTimeMillis();
              Car car1 = cglibProxy.getProxy(Bus.class);
              cglibproxyinitTime += System.currentTimeMillis() - startTime;

              startTime = System.currentTimeMillis();
              car1.drive();
              cglibInvokeTime += System.currentTimeMillis() - startTime;

          }
        }
        System.out.println("~~~~~~~~~~~~~~~~~~~~~Result~~~~~~~~~~~~~~~~~~~~~~~~~~");
        System.out.println("JDKInitTtime:" + jdkproxyInitTime);
        System.out.println("jdkInvokeTime:" + jdkInvokeTime);
        System.out.println("jdkInvokeTime:" + cglibproxyinitTime);
        System.out.println("cglibInvokeTime:" + cglibInvokeTime);
      }
    }
测试结果

相关文章

  • Java 动态代理

    java的动态代理机制详解 JDK动态代理详解 Java核心技术点之动态代理

  • JDK动态代理详解

    JDK动态代理详解 java动态代理类 Java动态代理类位于java.lang.reflect包下,一般主要涉及...

  • Java动态代理从入门到原理再到实战

    目录 前言 什么是动态代理,和静态代理有什么区别 Java动态代理的简单使用 Java动态代理的原理解读 动态代理...

  • java反射和动态代理

    java动态代理Proxy.newProxyInstance 详解java代理机制(静态代理、动态代理)以及使用场景

  • 保存java 动态代理生成的字节码文件

    保存java 动态代理生成的字节码文件 在Java中,常用的动态代理技术有JDK的动态代理和cglib动态代理,不...

  • java动态代理(JDK和cglib)(转载自http://ww

    java动态代理(JDK和cglib) JAVA的动态代理 代理模式 代理模式是常用的java设计模式,他的特征是...

  • java基础巩固笔记(4)-代理

    标签: java Contents java基础巩固笔记(4)-代理概念动态代理创建动态类动态代理的工作原理面向切...

  • java 动态代理

    动态代理动态代理可以让我们在运行时动态生成代理类,解耦程度更高。Java 动态代理的实现主要借助于 java.la...

  • java随笔(十一)

    java动态代理源码分析,总结。java动态代理实现步骤: 通过阅读源码发现,动态生成代理对象$Proxy0,该对...

  • Java基础:反射

    反射注解动态代理相关阅读 Java基础:类加载器 Java基础:反射 Java基础:注解 Java基础:动态代理 ...

网友评论

      本文标题:JAVA动态代理

      本文链接:https://www.haomeiwen.com/subject/icvmbttx.html