海澜之家网站建设的计划,门户网站建设方案ppt 百度文库,校园门户网站建设公司,小程序首页模板Java的两种动态代理方式动态代理是什么#xff1f;JDK动态代理CGLib动态代理CGLib 底层原理CGLib 实现步骤两者区别Spring AOP原理--动态代理动态代理是什么#xff1f; 动态代理就是#xff0c;在程序运行期#xff0c;创建目标对象的代理对象#xff0c;并对目标对象中的…
Java的两种动态代理方式动态代理是什么JDK动态代理CGLib动态代理CGLib 底层原理CGLib 实现步骤两者区别Spring AOP原理--动态代理动态代理是什么 动态代理就是在程序运行期创建目标对象的代理对象并对目标对象中的方法进行功能性增强的一种技术。 在生成代理对象的过程中目标对象不变代理对象中的方法是目标对象方法的增强方法。可以理解为运行期间对象中方法的动态拦截在拦截方法的前后执行功能操作。
代理类在程序运行期间创建的代理对象称之为动态代理对象。 这种情况下创建的代理对象并不是事先在Java代码中定义好的。而是在运行期间根据我们在动态代理对象中的“指示”动态生成的。也就是说你想获取哪个对象的代理动态代理就会为你动态的生成这个对象的代理对象。动态代理可以对被代理对象的方法进行功能增强。有了动态代理的技术那么就可以在不修改方法源码的情况下增强被代理对象的方法的功能在方法执行前后做任何你想做的事情。 创建代理对象的两个方法 JDK动态代理 Proxy.newProxyInstance(三个参数); CGLib动态代理 Enhancer.create(两个参数); JDK动态代理
基于接口的动态代理 提供者JDK 使用JDK官方的Proxy类创建代理对象 注意代理的目标对象必须实现接口
public class LogJdkProxy {/*** 生成对象的代理对象对被代理对象进行所有方法日志增强* 参数原始对象* 返回值被代理的对象* JDK 动态代理* 基于接口的动态代理* 被代理类必须实现接口* JDK提供的*/public static Object getObject(final Object obj){/*** 创建对象的代理对象* 参数一类加载器* 参数二对象的接口* 参数三调用处理器代理对象中的方法被调用都会在执行方法。对所有被代理对象的方法进行拦截*/Object proxyInstance Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), new InvocationHandler() {public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//方法执行前long startTime System.currentTimeMillis();Object result method.invoke(obj, args);//执行方法的调用//方法执行后long endTime System.currentTimeMillis();SimpleDateFormat sdf new SimpleDateFormat();System.out.printf(String.format(%s方法执行结束时间%%s 方法执行耗时%%d%%n, method.getName()), sdf.format(endTime), endTime - startTime);return result;}});return proxyInstance;}
}
CGLib动态代理
基于类的动态代理 提供者第三方 CGLib 使用CGLib的Enhancer类创建代理对象 注意如果报 asmxxxx 异常需要导入 asm.jar包
public class LogCGLibProxy {/*** 生成对象的代理对象对被代理对象进行所有方法日志增强* 参数原始对象* 返回值被代理的对象* 使用CGLib创建动态代理对象* 第三方提供的的创建代理对象的方式CGLib* 被代理对象不能用final修饰* 使用的是Enhancer类创建代理对象*/public static Object getObjectByCGLib(final Object obj){/*** 使用CGLib的Enhancer创建代理对象* 参数一对象的字节码文件* 参数二方法的拦截器*/Object proxyObj Enhancer.create(obj.getClass(), new MethodInterceptor() {public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {//方法执行前long startTime System.currentTimeMillis();Object invokeObject method.invoke(obj, objects);//执行方法的调用//方法执行后long endTime System.currentTimeMillis();SimpleDateFormat sdf new SimpleDateFormat();System.out.printf(String.format(%s方法执行结束时间%%s 方法执行耗时%%d%%n, method.getName()), sdf.format(endTime), endTime - startTime);return invokeObject;}});return proxyObj;}
}
CGLib 底层原理 通过查看 Enhancer 类源码最终也是生成动态代理类的字节码动态代理类继承要被代理的类然后实现其方法。 和 JDK Proxy 的实现代码比较类似都是通过实现代理器的接口再调用某一个方法完成动态代理的唯一不同的是CGLib 在初始化被代理类时是通过 Enhancer 对象把代理对象设置为被代理类的子类来实现动态代理的。
CGLib 实现步骤
创建一个实现接口 MethodInterceptor 的代理类重写 intercept 方法 创建获取被代理类的方法 getInstance(Object target); 获取代理类通过代理调用方法。
两者区别
JDK Proxy 和 CGLib 的区别主要体现在以下方面
JDK Proxy 是 Java 语言自带的功能无需通过加载第三方类实现Java 对 JDK Proxy 提供了稳定的支持并且会持续的升级和更新Java 8 版本中的 JDK Proxy 性能相比于之前版本提升了很多JDK Proxy是通过拦截器加反射的方式实现的 JDK Proxy 只能代理实现接口的类 JDK Proxy 实现和调用起来比较简单CGLib 是第三方提供的工具基于 ASM 实现的性能比较高CGLib 无需通过接口来实现它是针对类实现代理主要是对指定的类生成一个子类它是通过实现子类的方式来完成调用的。
Spring AOP原理–动态代理
1.AOP 思想 基于动态代理思想对原来目标对象创建代理对象在不修改原对象代码情况下通过代理对象调用增强功能的代码从而对原有业务方法进行增强。
2.AOP 作用 在不修改源代码的情况下可以增加额外的功能实现在原有功能基础上的增强。
3.AOP 实现原理 Spring AOP 的有两种实现方式JDK proxy 和 CGLib 动态代理
当 Bean 实现接口时Spring 使用 JDK proxy实现。 当 Bean 没有实现接口时Spring 使用 CGlib 代理实现。 通过配置可以强制使用 CGlib 代理(在 spring 配置中加入 aop:aspectj-autoproxy proxy-target-class“true”)。 4.AOP 使用场景
记录日志(调用方法后记录日志)监控性能(统计方法运行时间)权限控制(调用方法前校验是否有权限)事务管理(调用方法前开启事务调用方法后提交关闭事务 )缓存优化(第一次调用查询数据库将查询结果放入内存对象第二次调用直接从内存对象返回不需要查询数据库 )