zoukankan      html  css  js  c++  java
  • SpringBoot CGLIB AOP解决Spring事务,对象调用自己方法事务失效.

    对于像我这种喜欢滥用AOP的程序员,遇到坑也是习惯了,不仅仅是事务,其实只要脱离了Spring容器管理的所有对象,对于SpringAOP的注解都会失效,因为他们不是Spring容器的代理类,SpringAOP,就切入不了

    当然可以使用原生ASPECTJ,不用SpringAOP,但是基于生态链问题,还是尽量使用SpringAOP

    这里简单说一下,Spring如何选择使用CGLIB,或者是JDK代理,

    Spring 代理对象,cglib,jdk的问题思考,AOP 配置注解拦截 的一些问题.为什么不要注解在接口,以及抽象方法.

    简单来说,如果实现了某个接口,那么Spring就选择JDK代理(不一定),如果没有,那么就选择CGLIB代理,说起来简单,Spring还会闹脾气,不代理呢

    点击:根据Spring代理对象获得真实对象

    一直以来比较多的情况是在Controller 调用Service 的方法,把事务直接在Service的方法上,妥妥的没问题,事务正常执行

    考虑以下问题: 同对象的方法B 调用自己的方法A,这里的事务将会失效(严格上来说,只要对方法A使用注解AOP均会失效),原因是因为这里的this.调用的并不是Spring的代理对象

        
    @Service
    public class ClassA{ @Transactional(propagation = Propagation.REQUIRES_NEW) public void methodA(){ } /** * 这里调用methodA() 的事务将会失效 */ public void methodB(){ this.methodA(); } }

     最简单的解决方法为:(代理模式为JDK 的情况下(根据接口代理))

     @Service
        public class ClassA extends BaseClass{
    
            @Transactional(propagation = Propagation.REQUIRES_NEW)
        @Override
    public void methodA(){ } /** * 这里调用methodA() 的事务将会失效 */
         public void methodB(){         //使用getBean     ((BaseClass)SpringUtil.getBean("classA")).methodA(); } }

     第二种方式:

    解决第一步: 必须

    SpringBoot:注解开启cglib代理,开启 exposeProxy = true,暴露代理对象 (否则AopContext.currentProxy()) 会抛出异常

    @EnableAspectJAutoProxy(exposeProxy = true)
    public class Application{
        public static void main(String[] args) {
            SpringApplication.run(Application.class, args);
        }
    }

    配置方式:配置方式下,我用的是SpringBoot 1.5.4没有找到exposeProxy的选项,所以建议使用注解式.

    传统Spring配置文件

     XML

    <aop:aspectj-autoproxy expose-proxy="true"/>

    第二步(需要保证Spring对这个bean创建了代理对象,基本上涉及到Aop的方法的类,都会创建代理对象) 可以用以下代码判断

    /**
     * Created by laizhenwei on 19:37 2017-10-14
     */
    @Service("classImplProxy")
    @Scope(proxyMode = ScopedProxyMode.INTERFACES)
    public class ClassImplProxy implements IClass {
    
        @Override
        @Transactional
        public void sysout() {
        }
    
        //是否代理对象
        @Override
        public boolean isAopProxy() {
            return AopUtils.isAopProxy(AopContext.currentProxy());
        }
    
        //是否cglib 代理对象
        @Override
        public boolean isCglibProxy() {
            return AopUtils.isCglibProxy(AopContext.currentProxy());
        }
    
        //是否jdk动态代理
        @Override
        public boolean isJdkDynamicProxy() {
            return AopUtils.isJdkDynamicProxy(AopContext.currentProxy());
        }
    }

    获取代理对象    

      @Service
        public class ClassA{
    
            @Transactional(propagation = Propagation.REQUIRES_NEW)
            public void methodA(){
    
            }
    
            public void methodB(){
                //手动获取此对象Spring的代理类
                ((ClassA)AopContext.currentProxy()).methodA();
    } }

    第三种方式

    直接在当前类@Autowire 或者@Resource 注入自己,然后用注入的bean 调用方法

  • 相关阅读:
    Elasticsearch未授权访问漏洞
    基于github+hexo搭建个人博客(window)
    PHP字符串offset取值特性
    IIS 7安装ModSecurity实现WAF功能
    PHP代码审计笔记--代码执行漏洞
    使用cmi工具连接服务器远程装机exsi
    shell编程系列25--shell操作数据库实战之备份MySQL数据,并通过FTP将其传输到远端主机
    shell编程系列24--shell操作数据库实战之利用shell脚本将文本数据导入到mysql中
    shell编程系列23--shell操作数据库实战之mysql命令参数详解
    shell编程系列22--shell操作数据库实战之shell脚本与MySQL数据库交互(增删改查)
  • 原文地址:https://www.cnblogs.com/sweetchildomine/p/6978037.html
Copyright © 2011-2022 走看看