一. 定義切點(diǎn)-pointcut
/**
* 定義若干切點(diǎn),切點(diǎn)是某個(gè)位置,在該位置的前后可以做些自定義邏輯
**/
public class SelfPointcuts{
//切點(diǎn)1
@Pointcut("execution(public * com.lance.platform.controller.*.*(..))")
public void restController(){}
//切點(diǎn)2
@Pointcut("execution(public * com.lance.platform.controller2.*.*(..))")
public void restController2(){}
}
二. 定義切面-aspect
1. 切面定義
/**
* 切面是在切點(diǎn)的前后左右等定義相關(guān)邏輯的位置
**/
@Aspect
@Slf4j
public class SelfLogicAspect{
@Before("SelfPointcuts.restController()")
public void doSthBefore(JoinPoint joinpoint){
//自定義邏輯
log.info("before method.");
}
//restController2()雖不是static方法,但可以用“類.name()“形式調(diào)用
@Around("SelfPointcuts.restController2()")
public void doSthAround(ProceedingJoinPoint joinpoint){ //@Around時(shí),參數(shù)是ProceedingJoinPoint 類型
//自定義邏輯
log.info("before method.");
boolean flag = true;
if(flag) {
//驗(yàn)證通過,讓程序繼續(xù)執(zhí)行下去
joinpoint.proceed();
}else {
//做其他處理,一般中斷程序執(zhí)行,返回錯(cuò)誤信息給client
}
}
}
2. 自定義邏輯
案例1
//通過JoinPoint獲得,controller方法的各種信息
//方法modifier, 類package, 類名,方法名、方法參數(shù)、
org.aspectj.lang.Signature signature = joinPoint.getSignature()
signature.getName();
signature.getModifiers();
signature.getDeclaringType();
signature.getDeclaringTypeName();
//可以獲取請(qǐng)求request,進(jìn)而獲得對(duì)請(qǐng)求流程的控制
ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletRequest request = (HttpServletRequest) attributes.getRequest();
//獲取請(qǐng)求URL
String url = request.getRequestURL().toString();
//獲取請(qǐng)求方法 POST,PUT,GET,DELETE
String method = request.getMethod();
案例2
結(jié)合controller方法自定義注解做權(quán)限控制
//自定義注解,置到方法上
@Target({ElementType.TYPE, Elementtype.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface AuthResource{
String name(); //相當(dāng)于普通類中的 field
}
/**
* 定義controller,并用 自定義注解注解
**/
public class SelfController{
@AuthResource(name = "ONLY_READ")
@RequestMapping("/sth")
public String getSth() {
}
@AuthResource(name = "READ_AND_EDIT")
public String getSth() {
}
}
...
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
Method method = signature.getMethod();
AuthResource resource = method.getAnnotation(AuthResource.class);
//獲取 "ONLY_READ, READ_AND_EDIT" 等
String resourceName = resource.name();
//可以通過request 獲取當(dāng)前用戶名
String userName = (String) request.getAttribute("prawn.audit.username");
//DB中已經(jīng)分配過 user-role 對(duì)應(yīng)關(guān)系
//判斷該用戶是否有讀寫權(quán)限
.....
.....
...
3. AOP與Filter的對(duì)比及結(jié)合
1)執(zhí)行順序上 filter -> aop攔截
2)filter 可以執(zhí)行全局的控制,跟具體controller方法無關(guān),在進(jìn)入Controller之前過濾,一個(gè)粗粒度,一個(gè)細(xì)粒度到方法上
3)filter的創(chuàng)建及執(zhí)行請(qǐng)閱讀 ** 筆記