首页 > 其他分享 >AOP切入同类调用方法不起作用,AopContext.currentProxy()帮你解决这个坑

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

时间:2023-02-13 10:35:39浏览次数:34  
标签:user String void currentProxy AOP UserService public AopContext


原来在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>

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

 

 

 

在启动类上加上如下注解

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

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

 

新增一个实体类User

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

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接口

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

public interface UserService {
/**
* 模拟保存用户
* @param user
*/
void saveUser(User user);

/**
* 批量保存用户
* @param users
*/
void saveAllUser(List<User> users);
}

 

新增一个切面类

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

@Component
@Aspect//表明当前类是一个切面类
public class LogUtil {
/**
* 用于配置当前方法是一个前置通知
*/

@Before("execution(* com.example.demo1.demo.test3.*.saveUser(..))")
public void printLog() {
System.out.println("执行打印日志的功能");
}
}

 

 

 

新增UserServiceImpl实现类

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

@Service("userService")
public class UserServiceImpl implements UserService{
@Override
public void saveUser(User user) {
System.out.println("执行了保存用户"+user);
}

@Override
public void saveAllUser(List<User> users) {
for(User user : users){
// UserService proxyUserServiceImpl = (UserService) AopContext.currentProxy();
// proxyUserServiceImpl.saveUser(user);
saveUser(user);
}
}
}

 

新增测试类SpringEnableAspecctJAutoProxyTest

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

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

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

 

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

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

 

 

可以看到加上

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

执行打印日志的功能语句

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

标签:user,String,void,currentProxy,AOP,UserService,public,AopContext
From: https://blog.51cto.com/u_15258465/6053659

相关文章

  • 用注解实现AOP
    一、@Aspect:切面类@Before:前置@Around:环绕@AfterRetuming:后置@AfterThrowing:异常@After:最终@Pointcut:连接点,标注在一个无返、无参、二、加载<context:component-scanbase-p......
  • aop操作-AspectJ注解方式
    1.创建类,类中定义方法packagecom.xxx.spring.aop.aopanno;publicclassUser{publicvoidadd(){System.out.println("add...");}}2.创建增强......
  • aop术语和aop操作准备工作
    1.连接点类中哪些方法能被增强,这些方法被称为连接点2.切入点实际被真正增强的方法,称为切入点3.通知(增强)实际增强逻辑的部分,称为通知(增强):①前置通知②后置通知③环......
  • AOP基本概念
    一、面向切面编程(AspectOrientedProgramming)是对面向对象编程(OOP)的有益补充和升级AOP编程思想研究的不是层与层之间的关系,主要的是同一层的各个不同功能块共性功能,比如:访......
  • AOP开发流程
    一、导入并加载jar包AOPaopalliance-1.0.jaraspectjweaver-1.6.9.jarspring-aop-3.2.13.RELEASE.jarSpringspring-beans-3.2.13.RELEASE.jarspring-context-3.2.13.RELEASE.......
  • aop中的一些对象
    AOP编程过程中的Signature接口转载:https://blog.csdn.net/M_amazing/article/details/121747188此接口通常用于跟踪或记录应用程序以获取有关连接点的反射信息////Sou......
  • .Net6对AOP的多种支持之IAsyncResourceFilter
     环境:.Net6Web项目Mvc结构开发工具:VS2022IAsyncResourceFilter(资源缓存异步)IAsyncResourceFilter扩展   ASP.NETCore6提供的是接口IAsyncResourceFilter......
  • Springboot整合AOP和注解,实现丰富的切面功能
    简介我们在文章《SpringAOP与AspectJ的对比及应用》介绍了AOP的使用,这篇文章讲解一下AOP与注解的整合,通过注解来使用AOP,会非常方便。为了简便,我们还是来实现一个计时的功......
  • Spring AOP与AspectJ的对比及应用
    1简介AOP,即面向切面编程是很常用的技术,特别是在JavaWeb开发中。而最流行的AOP框架分别是SpringAOP和AspectJ。2SpringAOPvsAspectJSpringAOP是基于SpringIoC实......
  • Spring23 - 基于XML配置的AOP
    基于XML的AOP准备工作参考基于注解的AOP环境实现在.xml文件中对AOP进行配置<context:component-scanbase-package="com.atguigu.aop.xml"></context:component-s......