Spring动态代理的两种方式

Spring动态代理的两种方式,第1张

Spring动态代理的两种方式
  • 1. Spring创建的动态代理类在哪⾥?
    • 1.1 什么叫动态字节码技术?
  • 2.JDK的动态代理
    • 2.1 Proxy.newProxyInstance⽅法参数详解
    • 2.1 编码
  • 3. CGlib的动态代理
    • 3.1 编码
  • 4. 总结

1. Spring创建的动态代理类在哪⾥?

Spring 框架在运行时,通过字节码技术,在jvm创建的,运营在jvm内部,等程序结束后,会和jvm一起消失。

1.1 什么叫动态字节码技术?

通过第三方动态字节码框架,在jvm中创建对应类的字节码,进而创建对象,当虚拟机结束,动态字节码跟着消失。==
结论
动态代理不需要定义类文件,都是在jvm运行过程中创建的,所以不会造成静态代理,类文件数量过多,影响项目管理的问题。
第三方动态字节码框架:ASM,javassist ,cglib

2.JDK的动态代理 2.1 Proxy.newProxyInstance⽅法参数详解
Proxy.newProxyInstance(ClassLoader, Interfaces, handler);

参数1ClassLoader:借用一个类加载器,创建代理类,的class对象,进而可以创建代理对象。
参数2Interfaces:原始对象的接口,例如:

TestService testService = new TestServiceImpl();
//参数2Interfaces:testService.getClass().getInterfaces();

参数3:handler 即 invocationHandler
作用:用于书写额外功能,额外功能运行在原始方法之前 ,之后,抛出异常。
例如:

 InvocationHandler handler = new InvocationHandler() {
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("------proxy  log  before --------");
        //原始方法运行
        Object ret = method.invoke(testService, args);
        System.out.println("------proxy  log  after --------");
        return ret;
    }
};

上述代码中各个参数的含义:
返回值:Object:原始方法的返回值。
参数:proxy: 可忽略,代表的是代理对象;
method:额外功能所增加给的那个原始方法,可调用invoke()方法执行原始方法
args[]:原始方法的参数

2.1 编码

前置条件:
一个接口和一个接口的实现类,例如:
接口:

public interface TestService {

    boolean login(String username, String passoword);
}

实现类:

public class TestServiceImpl implements TestService {
    @Override
    public boolean login(String username, String passoword) {
        System.out.println("username = " + username + "password = " + passoword);
        return true;
    }
}

开始编写动态代理:

public class TestProxy {
 public static void main(String[] args) {
     //1. 创建原始对象
     TestService testService = new TestServiceImpl();
     //2. 创建JDK动态代理
     InvocationHandler handler = new InvocationHandler() {
         @Override
         public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
             System.out.println("------proxy  log  before --------");
             //原始方法运行
             Object ret = method.invoke(testService, args);
             System.out.println("------proxy  log  after --------");
             return ret;
         }
     };

     /*
       ClassLoader loader,借用一个类加载器 创建代理类的class对象  ,进而可以创建代理对象
       Class[] interfaces,
       InvocationHandler h
      */
     TestService testServiceProxy = (TestService) Proxy.newProxyInstance(TestServiceImpl.class.getClassLoader(), testService.getClass().getInterfaces(), handler);
     testServiceProxy.login("123", "123");
 }

运行结果:

3. CGlib的动态代理

CGlib创建动态代理的原理:⽗⼦继承关系创建代理对象,原始类作为⽗类,代理类作为⼦类,这样既可以保证2者⽅法⼀致,同时在代理类中提供新的实现(额外功能+原始⽅法);

3.1 编码

前置条件跟上面一样

public class TestCglib {
    public static void main(String[] args) {
        // 1. 创建原始对象
        TestService testService = new TestService();
        //2 .通过cglib创建动态代理对象
        Enhancer enhancer = new Enhancer();
        enhancer.setClassLoader(testService.getClass().getClassLoader());
        enhancer.setSuperclass(testService.getClass());
        MethodInterceptor methodInterceptor = (o, method, args1, methodProxy) -> {
            //等同于 InvocationHandler ---invoke
            System.out.println(" log= ");
            Object invoke = method.invoke(testService, args1);
            return invoke;
        };

        enhancer.setCallback(methodInterceptor);

        TestService o = (TestService) enhancer.create();
        o.login("yjy", "123");
    }
}

运行结果:

4. 总结
  1. JDK动态代理 Proxy.newProxyInstance() 通过接⼝创建代理的实现类
  2. Cglib动态代理 Enhancer 通过继承⽗类创建的代理类

若有错误,希望大佬指出。

对你有帮助给点个👍再走呗。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/737434.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-04-28
下一篇 2022-04-28

发表评论

登录后才能评论

评论列表(0条)

保存