系列文章
像MyBatis,OpenFeign那样自定义注解——扫描篇
像MyBatis,OpenFeign那样自定义注解——Jdk动态代理篇
像MyBatis,OpenFeign那样自定义注解——CGLib动态代理篇
像MyBatis,OpenFeign那样自定义注解——SpringAOP实现篇
项目地址:AnnotationDemo
在上一篇文章中,我们讲解了使用Jdk动态代理方式代理我们的接口,本篇文章将讲解CGLib动态代理的方式代理接口
按照惯例,先展示相关类:
CGLibHandler——CGlib代理类处理程序,实现MethodInterceptor接口
与Jdk代理一样,创建一个实现MethodInterceptor接口处理程序类CGLibHandler:
/**
* CGlib代理类处理程序
*
* @author 92058
*/
public class CGLibHandler implements MethodInterceptor {
/**
* @param o 代理对象
* @param method 代理方法
* @param objects 代理方法形惨
* @param methodProxy 被代理对象
* @return 当前调用方法所返回的内容
*/
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
return "调用了" + method.getName() + "方法" +
",参数" + Arrays.toString(objects) +
",方法上的注解:" + Arrays.toString(Arrays.stream(method.getAnnotations())
.map(annotation -> annotation.annotationType().getSimpleName())
.toArray());
}
}
让我们先回到上一篇文章中的registerBean()方法,找到下面代码:
//创建jdk代理处理程序
JdkHandler jdkHandler = new JdkHandler();
/*
* 生成代理对象
* 第一个参数传入一个ClassLoader
* 第二个参数传入被代理的接口,是一个数组,
* 这意味着你可以用一个代理对象去实现多个接口
* 第三个参数是代理处理程序
*/
Object object = Proxy.newProxyInstance(aClass.getClassLoader(),
new Class[]{aClass},
jdkHandler);
可以看到,我们在使用Jdk代理的时候首先创建了一个处理程序,然后创建一个与处理程序相关联的对象。CGLib也类似,也需要一个处理程序,然后生成一个与处理程序相关联的对象。直接把上面的代码替换成下面代码:
//创建CGLib代理程序
CGLibHandler cgLibHandler = new CGLibHandler();
//创建增强对象
Enhancer enhancer = new Enhancer();
//设置增强对象所继承的类
enhancer.setSuperclass(aClass);
//设置控制类
enhancer.setCallback(cgLibHandler);
//创建代理对象
Object object = enhancer.create();
至此,CGLib代理方式就完成了。
前面我用了3篇的文章介绍了如何实现类似MyBatis,OpenFeign那样的注解,先扫描到注解的接口,通过动态代理生成对象,并把对象注册成Bean,下一篇文章我们回到原点,讲解如何使用Spring AOP实现类似功能