Aspectj中@args使用误解

本文通过一个具体的AOP案例介绍了Spring AOP中@args注解的正确使用方式,澄清了初学者常见的误区,并分享了作者从实践中获得的经验。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

刚刚开始触AOP,什么都不懂就要做东西。在网上看例子,发现别人在切面类上用@annotation加自定义注解就可以拦截使用这个注解的方法。后来看到还有个@args,就以为可以自定义一个参数注解,在方法的参数上使用这个注解后切面可以拦截有这个参数的方法。由是写下如下的代码:

CheckPara.java

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface CheckPara {
	public int sub() default 0;
}

UserService.java

@Service("userService")

public class UserService {
	public void addUser(@CheckPara int age)
	{
		System.out.println("age: " + age);
	}
}
SubParaHandler.java

@Component
@Aspect
public class SubAgeHandler {

	@Before("@args(com.zjs.cglib.test.CheckPara)")
	public void subIt() throws Throwable {
		System.out.println("fuck!");
		return;
	}
}

主程序

public class App 
{
    public static void main( String[] args )
    {
        ApplicationContext ac = new ClassPathXmlApplicationContext("testBeans.xml");
        UserService us = (UserService) ac.getBean("userService");
        us.addUser(us);
    }
}

运行后发现fuck!没有被打印出来。百度很久无解,只好开VPN去Google国外搜搜。发现http://stackoverflow.com/questions/20121126/spring-aop-args-advice-refuses-to-run,这哥们和我遇到的问题一样。

下面有一个回答给了另一个链接http://stackoverflow.com/questions/16810617/aspectj-java-instrumentation-to-intercept-an-annoted-parameter-call-usage,进去之后第一个回答解决了问题。

原来这种写法AspectJ现在还不支持,是一个wish list上的feature,但还没有实现。

现在@args的正确用法是,自定义一个ElementType.TYPE的注解,这个注解用来修饰自定义类型(比如自己写的一个类),一个方法以这个自定义的类的实例为参数且只能有这唯一一参数,那这个方法在调用时会被匹配@args(自定义注解)的切面拦截。

这次的教训是。。真有问题的时候还必须得google。。

package com.ruoyi.framework.aspectj; import com.ruoyi.common.core.domain.BaseEntity; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.aspectj.lang.reflect.MethodSignature; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.stereotype.Component; import com.ruoyi.common.annotation.OrderHandler; import org.springframework.web.context.request.RequestContextHolder; import org.springframework.web.context.request.ServletRequestAttributes; import javax.servlet.http.HttpServletRequest; import java.lang.reflect.Method; import java.util.ArrayList; import java.util.Arrays; import java.util.List; @Aspect @Component public class OrderHandlerAspect { private static final Logger log = LoggerFactory.getLogger(OrderHandlerAspect.class); @Pointcut("@annotation(orderHandler)") public void OrderHandlerPointcut(OrderHandler orderHandler) { // Pointcut 表达式 } @Before("@annotation(orderHandler)") public void beforeOrderHandler(JoinPoint joinPoint, OrderHandler orderHandler) { Object[] args = joinPoint.getArgs(); String tableName = orderHandler.tableName(); for (Object arg : args) { if (arg instanceof BaseEntity) { BaseEntity baseEntity = (BaseEntity) arg; String ordersStr = baseEntity.getOrders(); String propsStr = baseEntity.getProps(); // 如果 `orders` 或 `props` 为空,则不进行排序 if (ordersStr == null || propsStr == null || ordersStr.trim().isEmpty() || propsStr.trim().isEmpty()) { baseEntity.setOrderByClause(""); return; } List<String> orders = Arrays.asList(ordersStr.split(",")); List<String> props = Arrays.asList(propsStr.split(",")); if (props.size() != orders.size()) { throw new IllegalArgumentException("筛选条件长度不匹配"); } StringBuilder orderByClause = new StringBuilder("ORDER BY "); for (int i = 0; i < props.size(); i++) { String prop = props.get(i).trim(); String order = orders.get(i).trim().toUpperCase(); if (!("ASC".equals(order) || "DESC".equals(order))) { throw new IllegalArgumentException("Invalid sort order: " + order); } prop = camelToSnake(prop); // 拼接 SQL if (tableName != null && !tableName.trim().isEmpty()) { orderByClause.append(tableName).append(".").append(prop); } else { orderByClause.append(prop); } orderByClause.append(" ").append(order); if (i < props.size() - 1) { orderByClause.append(", "); } } // 设置到 `BaseEntity` baseEntity.setOrderByClause(orderByClause.toString()); } } } /** * 将驼峰命名(camelCase)转换为下划线命名(snake_case) */ private String camelToSnake(String camelCase) { if (camelCase == null || camelCase.isEmpty()) { return camelCase; } return camelCase.replaceAll("([a-z])([A-Z])", "$1_$2").toLowerCase(); } } 在其中添加ALLOWED_COLUMNS部分,并且进行错误异常的处理部分 参考以下 // **动态获取实体类字段名** Set<String> allowedColumns = getEntityFields(entityClass); // **确保字段在实体类中存在** if (!allowedColumns.contains(prop)) { throw new IllegalArgumentException("非法排序字段: " + prop); } arg中包含所有的实体类的属性字段
最新发布
03-15
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值