Spring是经过动态代理来实现AOP的,它有2种实现动态代理机制。java
关于JDK动态代理以前已经写过了,这里不在讨论。spring
CGLib采用了很是底层的字节码技术,其原理是经过字节码技术为一个类建立子类,并在子类中采用方法拦截的技术拦截全部父类方法的调用,顺势织入横切逻辑。ide
Cglib代理,也称做子类代理,它是在内存中构建一个子类对象从而实现对目标对象功能的扩展。工具
Spring核心包中已经集成了Cglib,不须要另外导包。性能
package cn.iborder.factory; import java.lang.reflect.Method; import org.springframework.cglib.proxy.Enhancer; import org.springframework.cglib.proxy.MethodInterceptor; import org.springframework.cglib.proxy.MethodProxy; import cn.iborder.Dao.IUserDao; import cn.iborder.Dao.impl.UserDao; /** * 实现Cglib的MethodInterceptor接口 * @author iborder */ public class CglibProxyFactory implements MethodInterceptor { private Object target; //目标对象 public CglibProxyFactory(Object target) { this.target = target; } /** * 返回代理对象 */ public Object getProxyInstance() { //建立工具类 Enhancer enhancer = new Enhancer(); //设置父类 enhancer.setSuperclass(target.getClass()); //设置回调方法 enhancer.setCallback(this); //建立代理对象(子类) return enhancer.create(); } /** * 回调方法 */ @Override public Object intercept(Object object, Method method, Object[] args, MethodProxy proxy) throws Throwable { // TODO Auto-generated method stub System.out.println(this.getClass().getSimpleName()+"代理对象附加的操做,执行前"); //mehtod表明目标对象被调用的方法名 System.out.println(method.getName()); // 执行目标对象方法 Object result = method.invoke(target, args); System.out.println(this.getClass().getSimpleName()+"代理对象附加的操做,执行后"); return result; } public static void main(String[] args) { IUserDao target = new UserDao(); IUserDao userDao = (IUserDao) new CglibProxyFactory(target).getProxyInstance(); userDao.findUsers(); } }
CGLib建立的动态代理对象性能比JDK建立的动态代理对象的性能高很多,可是CGLib在建立代理对象时所花费的时间却比JDK多得多,因此对于单例的对象,由于无需频繁建立对象,用CGLib合适,反之,使用JDK方式要更为合适一些。同时,因为CGLib因为是采用动态建立子类的方法,对于final方法,没法进行代理!this