美文网首页
Spring AOP 与 动态代理

Spring AOP 与 动态代理

作者: 自负的鱼 | 来源:发表于2018-05-29 10:11 被阅读23次

    Spring AOP原理

    AOP作为面向对象的编程的一种补充,目前在系统中得到广泛的应用,例如在日志收集、权限管理、缓存、事务管理等方面。AOP代理分为静态代理和动态代理两种,其中静态代理是指使用AOP提供的编译命令在编译期进行代码增强;动态代理是指Spring在代码运行期动态的借助CJLIB、JDK动态代理生成动态代理类。

    使用AspectJ在编译时AOP

    AspectJ是Java语言AOP的一个实现,定义了如何表达、定义了AOP编程规范;另一部分包含了编译工具、调试工具等。
    AspectJ的静态代理他会在编译阶段将Aspect织入字节码中,运行的时候就是执行经过AOP增强的对象了。

    Spring AOP动态代理

    与AspectJ静态代理不一样,Spring AOP是动态代理。动态代理并不会修改字节码,而是为AOP对象在内存中临时生成一个代理对象,这个对象包含AOP对象的全部方法并在特定切点做增强处理,并回调原方法。

    Spring 允许使用AspectJ Annotation用于定义切面(Aspect)、切点(Pointcut)、增强处理(Advice)。
    Spring AOP 的 AfterReturning、Around 两种增强处理,但实际上 Spring 还支持 Before、After、AfterThrowing 等增强处理。

    Spring AOP代理是由IOC容器生成和管理,代理对象=目标对象+增强处理。
    动态代理在运行时通过反射创建一个实现了给定的一组接口的新类,该类实现的所有接口调用都会转到InvocationHandler.invoke()方法中执行。

    JDK 动态代理

    创建InvocationHandler执行功能增强

    public class CarTicketInvocation implements InvocationHandler{
    
        private Object target;
    
        public CarTicketInvocation(Object target){
            this.target = target;
        }
    
        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("i am prox invocation");
            return method.invoke(target,args);
        }
    }
    

    使用Proxy代理类工具运行时生成代理

    public static void main(String[] args) {
            ICarTicketPurchase ticketPurchase = new CarTicketPurchaseImpl();
            
            InvocationHandler handler = new CarTicketInvocation(ticketPurchase);
            ICarTicketPurchase prox = (ICarTicketPurchase)Proxy.newProxyInstance(ICarTicketPurchase.class.getClassLoader(), new Class[]{ICarTicketPurchase.class},handler);
    
            prox.purchase(10);
        }
    

    CJLIB代理

    CJLIB代理通过构造目标对象子类,从而实现对目标对象扩展完成动态代理。CJLIB底层使用字节码处理框架ASM实现对字节码转换生成子类。需要实现MethodInterceptor接口,在intercept 做功能增强。

    创建MethodInterceptor 实现功能增强

    public class CarTicketInterceptor implements MethodInterceptor{
    
        private Object target;
    
        public CarTicketInterceptor(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 obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
            System.out.println("i am prox interceptor");
            // 使用cglib 生成子类
            Object result = proxy.invokeSuper(obj,args);
    
            return result;
        }
    }
    

    生成子类代理类CJLIB

    public static void main(String[] args) {
            ICarTicketPurchase ticketPurchase = new CarTicketPurchaseImpl();
    
            ICarTicketPurchase proxy = (ICarTicketPurchase) new CarTicketInterceptor(ticketPurchase).getProxyInstance();
    
            proxy.purchase(10);
    
        }
    

    相关文章

      网友评论

          本文标题:Spring AOP 与 动态代理

          本文链接:https://www.haomeiwen.com/subject/cryojftx.html