开发中遇到一个统计用户使用次数的需求,例如,统计系统中搜索功能被用了几次,计算器功能被用了几次等等…
原本是想着在每个对应的方法中向数据库添加一条记录数据,但是后来朋友建议我使SpringAOP技术实现.
细想一下,直接在要统计的目标方法中修改代码不是很好,而且要统计的地方不止一处,这需要在每个需要统计的方法中修改代码,不仅代码重复,而且容易出错
最后想了想,还是打算使用aop实现吧
由于对aop不是很熟练,所以这么一个简单的功能磕磕碰碰,遇到了许多问题,特此记录一下,也希望对其他程序员朋友有所帮助.
小弟菜鸟,大神勿喷
- 在Spring中使用aop只需使用@Aspect注解声明这个Bean是一个切面,就可以在类中定义通知方法…
- 需要在spring-mvc.xml的配置文件中启用Spring对基于@AspectJ aspects的配置支持
- 注意使用spring4.3.8版本的时候aop:aspectj-autoproxy标签中需要添加proxy-target-class属性为true.否则无法拦截使用自定义注解的切点
- 没加这个配置,开始的时候发现方法怎么也没有被拦截
<!-- 启用Spring对基于@AspectJ aspects的配置支持 -->
<aop:aspectj-autoproxy proxy-target-class="true"/>
- 自定义一个注解标签,在需要统计的方法是添加这个注解,这样不会修改原有的方法
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface UserUsedCountTag {
int projectType() default -1;
int usedType() default -1;
int equipment() default -1;
}
- 自定义注解使用@Target表明,
- ElementType.METHOD表示该注解应用在类方法上
- 注解中可以定义类参数并设置默认值
- 在需要统计的方法是添加该注解
- 定义一个切面类,拦截使用该注解的方法
import com.alibaba.fastjson.JSONObject;
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.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import javax.servlet.http.HttpServletRequest;
@Aspect
@Component
public class UserUsedCountAspect {
// 切入点为 使用UserUsedCountTag 注解的任何地方
@Pointcut("@annotation(com.xie.mao.shu.UserUsedCountTag)")
public void userUsedCountPoint(){}
/**
* 方法前置通知
* 在目标方法执行前
* @param joinPoint
*/
@Before(value = "userUsedCountPoint() && @annotation(userUsedCountTag)")
public void before(JoinPoint joinPoint,UserUsedCountTag userUsedCountTag){
// 自定义的处理程序
}
}
- 在通知方法上指明拦截的方法有使用userUsedCountTag注解,于是可以在通知方法中加入注解类型的参数
- 有了注解类型的参数,就可以在通知方法中获取到方法注解上定义的参数值
- 获取方式为:注解参数名称.注解属性名()
- 这样只需要在需要统计的地方添加 @UserUsedCountTag 这个注解就可以实现统计功能了,而且注解中可以自定义添加属性,可以用来区分不同的操作类型,这样在统计的时候也可以统计出不同的类型.