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

原来在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);代码后输出了切面类中的

执行打印日志的功能语句

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/269869.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[转]Installing Memcached on Windows

Installing Memcached on Windows 原文链接https://commaster.net/content/installing-memcached-windowsSubmitted by COMMASTER21JAN 15Memcached is a high performance, in-memory key-value store or caching system. Its main purpose is to speed up web applications b…

Java高级面试题

Java多线程 1、线程池的原理&#xff0c;为什么要创建线程池&#xff1f; 答&#xff1a;1)线程池可以降低创建和销毁线程时的资源消耗&#xff0c;提高响应速度&#xff0c;提高现成的可管理性。 2)线程池构造参数&#xff1a; corePoolSize:核心线程数 maximumPoolSize:最大…

访问修改属性日志

1 import time as t2 3 class Record:4 def __init__(self,value None,name None):5 self.value value6 self.name name7 8 def __get__(self,instance,owner):9 with open(D://record.txt,a) as f: 10 f.write(%s变量于北京时…

不自定义异步方法的线程池默认使用SimpleAsyncTaskExecutor

如果不自定义异步方法的线程池默认使用SimpleAsyncTaskExecutor。SimpleAsyncTaskExecutor&#xff1a;不是真的线程池&#xff0c;这个类不重用线程&#xff0c;每次调用都会创建一个新的线程。并发大的时候会产生严重的性能问题。 定义通用线程池 EnableAsync Configuratio…

同步Android与PC的时间

同步Android与PC的时间 在做一些网络延迟测试的时候&#xff0c;需要同步Android设备或者模拟器与PC的时间&#xff08;要不然无法准确计算延迟&#xff09;&#xff0c;在这里记一下获取Android的时间戳以及MacOS的时间戳&#xff0c;均为纳秒级 Android: adb shell echo \$EP…

AopContext.currentProxy();为什么能获取到代理对象

在同一个类中&#xff0c;非事务方法A调用事务方法B&#xff0c;事务失效&#xff0c;得采用AopContext.currentProxy().xx()来进行调用&#xff0c;事务才能生效。 B方法被A调用&#xff0c;对B方法的切入失效&#xff0c;但加上AopContext.currentProxy()创建了代理类&#x…

java中 set,list,array(集合与数组)相互转换

1 public static Object[] List2Array(List<Object> oList) { 2 Object[] oArray oList.toArray(new Object[] {}); 3 // TODO 需要在用到的时候另外写方法&#xff0c;不支持泛型的Array. 4 return oArray; 5 } 6 7 publi…

@Async注解导致循环依赖,BeanCurrentlyInCreationException异常

使用Async异步注解导致该Bean在循环依赖时启动报BeanCurrentlyInCreationException异常的根本原因分析&#xff0c;以及提供解决方案 今天在自己项目中使用Async的时候&#xff0c;碰到了一个问题&#xff1a;Spring循环依赖&#xff08;circular reference&#xff09;问题。 …

人工智能:图像数字化相关的知识介绍

❤️作者主页&#xff1a;IT技术分享社区 ❤️作者简介&#xff1a;大家好,我是IT技术分享社区的博主&#xff0c;从事C#、Java开发九年&#xff0c;对数据库、C#、Java、前端、运维、电脑技巧等经验丰富。 ❤️个人荣誉&#xff1a; 数据库领域优质创作者&#x1f3c6;&#x…

《深入理解Java虚拟机》读书笔记

堆分配参数&#xff1a; -XX:PrintGC 使用该参数&#xff0c;虚拟机启动后&#xff0c;只要遇到GC就会打印日志&#xff1b; -XX&#xff1a;UseSerialGC 配置串行回收器&#xff1b; -XX&#xff1a;PrintGCDeltails 可以查看详细信息&#xff0c;包括各个区的情况 -Xms&#…

线程可见性和关键字volatile

线程可见性 可以看到程序变量running没volatile是死循环 加了volatile成功输出 public class VolitaleTest {private static volatile boolean running true;public static void main(String[] args) {Thread thread new Thread(() ->{long i 0L;while (running){i;}Sys…

codeforce 768B Code For 1

题意&#xff1a;将n分解为n/2, n%2, n/2三部分&#xff0c;再将n/2分解。。得到一个序列只有0和1&#xff0c;给出[l, r]问l到r有几个1#include <stdio.h> #define ll __int64 ll query(ll L,ll R,ll l,ll r,ll n){if(l r) return n;ll mid (lr)>>1, ans0;if(L …

每秒钟承载600万订单级别的无锁并行计算框架 Disruptor学习

1.来源 Disruptor是英国外汇交易公司LMAX开发的一个高性能队列&#xff0c;研发的初衷是解决内部的内存队列的延迟问题&#xff0c;而不是分布式队列。基于Disruptor开发的系统单线程能支撑每秒600万订单&#xff0c;2010年在QCon演讲后&#xff0c;获得了业界关注。 2.应用背…

logisim输出变成红色的e_新车实拍解析 福特Mustang Mach-E亮点实拍图解

福特Mustang Mach-E新车主要针对造型设计对外进行了首次亮相发布&#xff0c;对新车内饰以及具体新车方面的数据信息暂未公布。如果消费者想要了解这款新车&#xff0c;大家可以继续关注《杨总继续观察》带来这款新车的详细报道。新车在设计上可以看作是一款福特野马的电动跨界…

castle windsor学习----- Services and Components 两者的定义

转载于:https://www.cnblogs.com/lanpingwang/p/6534208.html

html5 接收蓝牙广播_蓝牙定位技术浅析(化工厂应用)

蓝牙定位基于RSSI(Received Signal Strength Indication&#xff0c;信号场强指示)定位原理。根据定位端的不同&#xff0c;蓝牙定位方式分为网络侧定位和终端侧定位。由于蓝牙由于是近场通信其定位精度取决于点位的部署密度&#xff0c;一般会设计成7-8米一个定位基站&#xf…

C#中的泛型和泛型集合

泛型 泛型引入了一个概念:类型参数。通过使用类型参数&#xff08;T&#xff09;减少了运行时强制转换或装箱操作的风险&#xff0c;通过泛型可以最大限度的重用代码&#xff0c;保护类型的安全及提高性能&#xff0c;他的最常见应用就是创建集合类&#xff0c;可以约束集合类中…