SpringBoot自定義路由覆蓋實現(xiàn)流程詳解
背景
公司最近有一個項目二期需要對一些功能進行改造,涉及部分框架內(nèi)置業(yè)務接口個性化定制,兼容老接口功能并且增加一部分新的數(shù)據(jù)返回,由于前端調(diào)用這些接口分布較多且較為零碎,修改測試成本較大,所以打算在框架層面提供路由覆蓋功能,加快項目進度減少無技術含量的修改帶來的系統(tǒng)風險
設計
- 提供自定義注解指定需要覆蓋的路由及新路由地址
- 系統(tǒng)啟動時掃描所有注解數(shù)據(jù)并進行映射處理
- 注冊自定義路由映射配置類
實現(xiàn)
注解定義
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface CoverRoute {
String value() default "";
}
注解掃描及管理
在系統(tǒng)啟動時調(diào)用initRoute方法,把原路由和對應的覆蓋路由映射到map鍵值對中
public class ConverRouteUtil {
private static HashMap<String, String> mappingRegist = new HashMap<>();
public static void initRoute(Class runtimeClass, List<String> extraPackageNameList) {
List<Class<?>> scanClassList = new ArrayList<>();
if (!runtimeClass.getPackage().getName().equals(Application.class.getPackage().getName())) {
scanClassList.addAll(ScanUtil.getAllClassByPackageName_Annotation(runtimeClass.getPackage(), CoverRoute.class));
}
for (String packageName : extraPackageNameList) {
scanClassList.addAll(ScanUtil.getAllClassByPackageName_Annotation(packageName, CoverRoute.class));
}
for (Class clazz : scanClassList) {
CoverRoute coverRoute = (CoverRoute) clazz.getAnnotation(CoverRoute.class);
if (StringUtil.isEmpty(coverRoute.value())) {
continue;
}
RequestMapping requestMapping = (RequestMapping) clazz.getAnnotation(RequestMapping.class);
String classRoute = "";
if (requestMapping != null) {
classRoute = requestMapping.value()[0];
} else {
continue;
}
List<Method> methodList = Arrays.asList(clazz.getDeclaredMethods());
for (Method method : methodList) {
PostMapping postMapping = method.getAnnotation(PostMapping.class);
String methodRoute = "";
if (postMapping != null) {
methodRoute = postMapping.value()[0];
} else {
GetMapping getMapping = method.getAnnotation(GetMapping.class);
if (getMapping != null) {
methodRoute = getMapping.value()[0];
}
}
if (!StringUtil.isEmpty(classRoute) && !StringUtil.isEmpty(methodRoute)) {
String orginalRoute = coverRoute.value() + methodRoute;
String redirectRoute = classRoute + methodRoute;
mappingRegist.put(orginalRoute, redirectRoute);
}
}
}
if (mappingRegist.size() > 0) {
System.out.println("掃描路由方法覆蓋:" + mappingRegist.size() + "個");
}
}
public static boolean checkExistCover(String orginalRoute) {
return mappingRegist.containsKey(orginalRoute);
}
public static String getRedirectRoute(String orginalRoute) {
return mappingRegist.get(orginalRoute);
}
}
自定義RequestMappingHandlerMapping
繼承RequestMappingHandlerMapping重寫lookupHandlerMethod方法,在spring進行路由尋址時進行覆蓋
public class CustomRequestMappingHandlerMapping extends RequestMappingHandlerMapping {
@Override
protected HandlerMethod lookupHandlerMethod(String lookupPath, HttpServletRequest request) throws Exception {
if(ConverRouteUtil.checkExistCover(lookupPath)){
String redirectRoute = ConverRouteUtil.getRedirectRoute(lookupPath);
request.setAttribute("redirectTag","1");
request.setAttribute("redirectRoute",redirectRoute);
request.setAttribute("lookupPath",lookupPath);
lookupPath = redirectRoute;
}else{
request.setAttribute("redirectTag","0");
}
return super.lookupHandlerMethod(lookupPath, request);
}
@Override
protected RequestMappingInfo getMatchingMapping(RequestMappingInfo info, HttpServletRequest request) {
String redirectTag = ConvertOp.convert2String(request.getAttribute("redirectTag"));
if(redirectTag.equals("1")){
String redirectRoute = ConvertOp.convert2String(request.getAttribute("redirectRoute"));
boolean check = false;
if( info.getPatternsCondition()!=null){
Set<String> set = info.getPatternsCondition().getPatterns();
if(set.size()>0){
String[] array = new String[set.size()];
array = set.toArray(array);
String pattern = array[0];
if(pattern.equals(redirectRoute)){
check = true;
}
}
}
if(check){
return info;
}else{
return super.getMatchingMapping(info, request);
}
}else{
return super.getMatchingMapping(info, request);
}
}
}
注冊RequestMappingHandlerMapping
@Component
public class WebRequestMappingConfig implements WebMvcRegistrations {
public RequestMappingHandlerMapping getRequestMappingHandlerMapping() {
RequestMappingHandlerMapping handlerMapping = new CustomRequestMappingHandlerMapping();
handlerMapping.setOrder(0);
return handlerMapping;
}
}
使用示例
在個性化接口類增加@CoverRoute注解,指定需要覆蓋的路由地址,創(chuàng)建相同路由路徑的的方法即可,訪問原來的接口地址會自動轉(zhuǎn)發(fā)到項目個性化接口地址
原接口
@Controller
@RequestMapping("/example/original")
public class RedirectOriginalExampleController {
@PostMapping("/getConfig")
@ResponseBody
@AnonymousAccess
public Object getConfig(@RequestBody Map<String, Object> params) {
Result result = Result.okResult();
result.add("tag","original");
return result;
}
}
新接口
@Controller
@RequestMapping("/example/redirect")
@CoverRoute("/example/original")
public class RedirectExampleController {
@PostMapping("/getConfig")
@ResponseBody
public Object getConfig(@RequestBody Map<String, Object> params) {
Result result = Result.okResult();
String param1 = ConvertOp.convert2String(params.get("param1"));
result.add("tag","redirect");
result.add("param1",param1);
return result;
}
}
到此這篇關于SpringBoot自定義路由覆蓋實現(xiàn)流程詳解的文章就介紹到這了,更多相關SpringBoot自定義路由覆蓋內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Java生成二維碼的兩種實現(xiàn)方式(基于Spring?Boot)
這篇文章主要給大家介紹了關于Java生成二維碼的兩種實現(xiàn)方式,文中的代碼基于Spring?Boot,本文基于JAVA環(huán)境,以SpringBoot框架為基礎開發(fā),文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2023-07-07
java 使用ImageIO.writer從BufferedImage生成jpeg圖像遇到問題總結(jié)及解決
這篇文章主要介紹了java 使用ImageIO.writer從BufferedImage生成jpeg圖像遇到問題總結(jié)及解決的相關資料,需要的朋友可以參考下2017-03-03
SpringBoot使用Kaptcha實現(xiàn)驗證碼的生成與驗證功能
這篇文章主要介紹了SpringBoot使用Kaptcha實現(xiàn)驗證碼的生成與驗證功能,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-03-03
Springboot實現(xiàn)動態(tài)定時任務流程詳解
通過重寫SchedulingConfigurer方法實現(xiàn)對定時任務的操作,單次執(zhí)行、停止、啟動三個主要的基本功能,動態(tài)的從數(shù)據(jù)庫中獲取配置的定時任務cron信息,通過反射的方式靈活定位到具體的類與方法中2022-09-09
java數(shù)據(jù)結(jié)構(gòu)排序算法之樹形選擇排序詳解
這篇文章主要介紹了java數(shù)據(jù)結(jié)構(gòu)排序算法之樹形選擇排序,結(jié)合具體實例形式分析了java樹形選擇排序的原理、實現(xiàn)技巧與相關注意事項,需要的朋友可以參考下2017-05-05

