jdk动态代理的局限性
首先我们知道,使用jdk创建动态代理有个局限,就是他只能为接口创建代理实例,对于那些没有实现接口的业务类,就不能使用jdk动态代理了;所以我们就需要使用CGLib动态代理。
CGLib动态代理
CGLib是一个强大高性能的代码生成库,被广泛应用于AOP框架,用其提供的方法进行拦截操作。
CGLib底层采用的是字节码技术,可以为被代理生成一个子类,而这个子类也就是代理对象,使用此代理对象调用父类中的方法时,都会被回调函数拦截,从而可以进行切面编程,增强方法。
CGLib动态代理的实现
主要使用的类Enhancer
增强器类,调用这个类的静态方法create(class clazz, Callback callback)
方法,即可创建执行类的代理对象
-
被代理对象(注意cglib动态代理方式要求被代理对象不能是接口的实现类)
/** * 模拟业务层 */ public class ServiceTest { public void demo() { System.out.println("我是Service类中的demo方法"); } }
-
编写cglib动态代理
public class CgProxy { public static void main(String[] args) { /* 使用Enhancer增强器的create()方法创建动态代理 参数1:被代理对象的字节码对象 参数2:一般创建Callback接口的子接口的实现类 MethodInterceptor -> 以匿名内部类的方式 */ ServiceTest serviceTest = (ServiceTest) Enhancer.create(ServiceTest.class, new MethodInterceptor() { /** * 拦截方法,每次使用代理对象,调用父类方法时,都会进入此回调方法。 * @param o 内部生成的代理对象 * @param method 拦截到的调用方法 * @param objects 拦截方法的参数 * @param methodProxy 当前执行方法的代理对象 * @return * @throws Throwable */ @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println("方法执行之前!"); //使用invoke()方法,参数为被代理对象的实例对象 Object result = method.invoke(ServiceTest.class.newInstance(), objects); System.out.println("方法执行之后!"); return result; } }); //调用方法 serviceTest.demo(); } }
-
结果