CGLxib动态代理
我们知道JDK动态代理必须实现接口,使用的反射方式效率也不是很高。
使用CGLib实现动态代理,完全不受代理类必须实现接口的限制,而且CGLib底层采用ASM字节码生成框架,使用字节码技术生成代理类,比使用Java反射效率要高。唯一需要注意的是,CGLib不能对声明为final的方法进行代理,因为CGLib原理是动态生成被代理类的子类。
-
需要一个被代理类
-
一个拦截器,这个拦截器是实现MethodInterceptor接口,在调用目标方法时候,cglib会回掉这个接口的intercept方法,实现你自己的逻辑
/** * * @param enhanced 由CGLib动态生成的代理类实例 * @param method 上文中实体类所调用的被代理的方法引用 * @param args 为参数值列表 * @param methodProxy 为生成的代理类对方法的代理引用 * @return * @throws Throwable */ public Object intercept(Object enhanced, Method method, Object[] args, MethodProxy methodProxy)
-
字节码增强器
Enhancer enhancer = new Enhancer(); //回调方法,就是实现了MethodInterceptor接口的子类 enhancer.setCallback(callback); //要生成代理对象的父类 enhancer.setSuperclass(Car.class); //生成代理,可以接收返回值 enhancer.create()
mybatis中的懒加载,假如对一个User的属性懒加载,那么会生成User的代理对象,通过user代理对象调用方法时,会去调用代理方法。
Javassist动态代理
javassist与JDK代理类似,它是通过代理工厂来创建代理对象的,代理类实现MethodHandler
接口,来拦截我们的接口请求,在invoke()
中做增强与请求转发处理。Javassist是一个开源的分析、编辑和创建Java字节码的类库。是一种字节码增强技术。
static Object crateProxy(Class<?> type, MethodHandler callback, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
// 创建 javassist ProxyFactory 对象
ProxyFactory enhancer = new ProxyFactory();
enhancer.setSuperclass(type);
enhanced = enhancer.create(typesArray, valuesArray);
//设置代理对象的执行器
((Proxy) enhanced).setHandler(callback);
MyBatis懒加载
懒加载也加延迟加载,主要以应用与**Mybatis**的关联查询,按照设置的延迟规则,推迟对延迟对关联对象的select查询,例如,我们在用Mybatis进行一对多的时候,先查询出一方,当程序需要多方数据时,mybatis会再次发出sql语句进行查询,减轻了对我们数据库的压力。Mybatis的延迟加载,只对关联对象有延迟设置。
Mybatis中对于延迟加载设置,只对于resultMap中的collection和association起作用,可以设置全局的,也可以单独设置,懒加载是在你java程序中调用该对象的子属性时,才会执行collection和association中的select方法。
加载时机
直接加载:执行完对主加载对象的 select 语句,马上执行对关联对象的 select 查询。
侵入式延迟: 执行对主加载对象的查询时,不会执行对关联对象的查询。但当要访问主加载对象的详情属性时,就会马上执行关联对象的select查询。
深度延迟: 执行对主加载对象的查询时,不会执行对关联对象的查询。访问主加载对象的详情时也不会执行关联对象的select查询。只有当真正访问关联对象的详情时,才会执行对关联对象的 select 查询。
<!--全局参数设置-->
<settings>
<!--延迟加载总开关-->
<setting name="lazyLoadingEnabled" value="true"/>
<!--侵入式延迟加载开关-->
<!--3.4.1版本之前默认是true,之后默认是false-->开启深度延迟加载,将value改为false即可
<setting name="aggressiveLazyLoading" value="true"/>为true会强行全部加载,为false则按需加载
</settings>
在单个resultMap中使用延迟加载
上面都是通过在mybatis.xml文件中统一配置的深度延迟加载,倘若只希望某些查询支持深度延迟加载的话可以在resultMap中的collection或association添加fetchType属性,配置为lazy之后是开启懒加载,配置eager是立即加载。fetchType属性将取代全局配置参数lazyLoadingEnabled的设置
<resultMap id="myua" type="ua"> <id property="addressId" column="aid"></id>
<collection fetchType="lazy" property="addresses" ofType="Address" column="aid" javaType="ArrayList" select="selectAddress"> </collection> </resultMap>
懒加载源码
在源码中可以发现,可以选择CGLib动态代理和Javassist动态代理(默认),当执行xml中方法时,执行完sql语句后,在ResultSetHandler中处理结果集。
#####若是有延迟加载的属性,则为这个对象创建代理对象