栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

Spring动态代理的两种方式

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

Spring动态代理的两种方式

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;
         }
     };

     
     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 通过继承⽗类创建的代理类

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

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

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/836846.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号