原来在springAOP的用法中,只有代理的类才会被切入,我们在controller层调用service的方法的时候,是可以被切入的,但是如果我们在service层 A方法中,调用B方法,切点切的是B方法,那么这时候是不会切入的,解决办法就是如上所示,在A方法中使用((Service)AopContext.currentProxy()).B() 来调用B方法,这样一来,就能切入了!

Spring的切面功能,是通过代理的方式来实现的,实现方式有JDK动态代理方式及Cglib的方式。

有时需要在service层通过AOP做一些日志、权限、监控等功能,但在service中进行自调用时,无法再次走进代理类中,因此会导致漏日志的情况出现。

此时,可通过AopContext.currentProxy()来解决问题。用AopContext.currentProxy().xxxx()的方式代替this.xxxx()的自调用方式。
AopContext.currentProxy()的本质是使用的ThreadLocal生成本地代理,这样的做法可能影响性能,后续文章对ThreadLocal的内部原理和性能进行进一步深入!

AopContext.currentProxy()该用法的意义

下面讲用一个例子讲解

首先加入依赖

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId><scope>test</scope></dependency><dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId></dependency>

在启动类上加上如下注解


@SpringBootApplication
@Configuration//@EnableAsync
@EnableAspectJAutoProxy(exposeProxy = true)//开启spring注解aop配置的支持

新增一个实体类User

public class User implements Serializable {private String id;private String username;private String password;private String email;private Date birthday;private String gender;private String mobile;private String nickname;public String getId() {return id;}public void setId(String id) {this.id = id;}public String getUsername() {return username;}public void setUsername(String username) {this.username = username;}public String getPassword() {return password;}public void setPassword(String password) {this.password = password;}public String getEmail() {return email;}public void setEmail(String email) {this.email = email;}public Date getBirthday() {return birthday;}public void setBirthday(Date birthday) {this.birthday = birthday;}public String getGender() {return gender;}public void setGender(String gender) {this.gender = gender;}public String getMobile() {return mobile;}public void setMobile(String mobile) {this.mobile = mobile;}public String getNickname() {return nickname;}public void setNickname(String nickname) {this.nickname = nickname;}}

新增UserService接口

public interface UserService {/*** 模拟保存用户* @param user*/void saveUser(User user);/*** 批量保存用户* @param users*/void saveAllUser(List<User> users);
}

新增一个切面类

@Component
@Aspect//表明当前类是一个切面类
public class LogUtil {/*** 用于配置当前方法是一个前置通知*/@Before("execution(* com.example.demo1.demo.test3.*.saveUser(..))")public void printLog() {System.out.println("执行打印日志的功能");}
}

新增UserServiceImpl实现类

@Service("userService")
public class UserServiceImpl implements UserService{@Overridepublic void saveUser(User user) {System.out.println("执行了保存用户"+user);}@Overridepublic void saveAllUser(List<User> users) {for(User user : users){
//            UserService proxyUserServiceImpl = (UserService) AopContext.currentProxy();
//            proxyUserServiceImpl.saveUser(user);saveUser(user);}}
}

新增测试类SpringEnableAspecctJAutoProxyTest

public class SpringEnableAspecctJAutoProxyTest {public static void main(String[] args) {//1.创建容器AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(DemoApplication.class);//2.获取对象UserService userService = ac.getBean("userService",UserService.class);//3.执行方法User user = new User();user.setId("1");user.setUsername("test");List<User> users = new ArrayList<>();users.add(user);userService.saveAllUser(users);}
}

启动测试类

输出

执行了保存用户com.example.demo1.demo.test3.User@5db99216

把注释的这两行代码放开,然后运行代码
UserService proxyUserServiceImpl = (UserService) AopContext.currentProxy();
proxyUserServiceImpl.saveUser(user);

可以看到加上

UserService proxyUserServiceImpl = (UserService) AopContext.currentProxy(); proxyUserServiceImpl.saveUser(user);代码后输出了切面类中的

执行打印日志的功能语句

AOP切入同类调用方法不起作用,AopContext.currentProxy()帮你解决这个坑相关推荐

  1. ❤️再也不用为了重写方法而苦恼了,Lombok帮你解决!

    ❤️再也不用为了重写方法而苦恼了,Lombok帮你一件解决问题 ​ 项目Lombok是一个java库,它可以自动插入到您的编辑器和构建工具中,为您的java注入活力. 永远不要再编写另一个getter ...

  2. java 同类调用方法_JAVA中同类中方法的调用问题?

    1.为什么可以直接new+构造方法,来创建对象.而不用创建实例引用呢,像常见的这种:Car car1=new Car;我记得看过有说java中没有被引用的对象将被GC处理.那这个算不算. 你创建了一个 ...

  3. Spring AOP 同类调用失效问题及解决方案

    首先我们看一下Spring Aop的原理图 在Spring中经常使用自定义注解或是spring已经封装 好的注解,通过AOP的方式是实现代码复用,避免重复劳动.而Spring实现AOP是通过动态代理来 ...

  4. springboot 调用方法事物_SpringBoot 内部方法调用,事务不起作用的原因及解决办法...

    在做业务开发时,遇到了一个事务不起作用的问题.大概流程是这样的,方法内部的定时任务调用了一个带事务的方法,失败后事务没有回滚.查阅资料后,问题得到解决,记录下来分享给大家. 场景 我在这里模拟一个场景 ...

  5. python实例方法不可以用类调用_python中可以直接用类调用方法吗

    python中可以直接用类调用方法吗 关注:244  答案:4  mip版 解决时间 2021-01-16 12:20 提问者 妳莂騙莪 2021-01-15 21:49 python中可以直接用类调 ...

  6. 探究Spring AOP 同类调用失效问题

    扯一下沟通 本来这篇文章是想写关于职场上的沟通的,尽管我也是一个初入职场的小白.但是发现时间不太够,只好作罢,后面再补上吧.关于沟通,我总结起来就是"沟通渠道要畅通,沟通要主动,沟通要高效& ...

  7. 在事务方法中调用另外一个事务方法,被调用方法的事务没起作用

    在事务方法中调用另外一个事务方法,被调用方法的事务没起作用 在事务方法中调用另外一个事务方法,被调用方法的事务没起作用 问题描述:(例) service层有两个事务方法insertUser1.inse ...

  8. 同类调用AOP失效问题

    使用aop代理对象,我们的到的原始对象实际上是一个他的代理对象 如: 结果: 这里可以得知,AOP是通过代理对象调用切入点(被通知的连接点),实现切面关系,AOP通知才会生效. 今天就疏忽了,连接点调 ...

  9. Spring AOP无法拦截内部方法调用-- expose-proxy=true用法

    假设一个接口里面有两个方法: package demo.long;public interface CustomerService { public void doSomething1(); publ ...

最新文章

  1. linux安装完redis之后log会在,Linux安装redis logstash
  2. InetAddress类的一些方法
  3. 反射机制源码及原理学习
  4. SQL SERVER中强制类型转换cast和convert的区别
  5. 服务器运行jar包日志怎么清理,docker 启动jar包,并将日志文件进行挂载
  6. oracle 日期6,EF 6与Oracle - 如何加入日期字段?
  7. 如何做Teams Bot的测试覆盖
  8. Doc2Vec训练相似文章识别模型
  9. JavaWeb前台异常处理
  10. SkinSharp用法
  11. 关于SQL数据库中的数据到如何导出--SQL2000导入到SQL2008中
  12. java字符相似_JAVA 获取两个字符串的相似度
  13. JavaScript的eval()函数(1)
  14. Windows小功能合集
  15. Excel表格数据生成sql插入语句
  16. html导航栏自动调间距,HTML CSS导航栏间距
  17. 三目运算(三目表达式)是什么?
  18. CSP多USBkey操作获取信息
  19. U盘不能分区,不能格式化,解决方案
  20. 【嵌入式】——串口实验——实现芯片串口收发数据,按键中断串口发送数据,串口接收数据中断来控制LED亮/灭

热门文章

  1. js function理解
  2. Oracle SQL篇(三)Oracle ROWNUM 与TOP N分析
  3. [LeetCode]题解(python):058-Length of Last Word
  4. 使用git如何批量对文件进行rm操作
  5. 修改Advance Template Jsp模板的编码格式
  6. 转载:https://blog.csdn.net/dcrmg/article/details/52939318
  7. 快速上手SpyGlass——基本流程
  8. Gmapping——从原理到实践
  9. 圆形卡尺测量后创建模板
  10. css 横线_atom.css正式发布,从此跟CSS框架说拜拜。