如何在SpringBoot项目中灵活配置自定义拦截接口路径

假如现在需要拦截例如以 /zjdl/huz/performance/board 开头的接口

1. 在 ResponseBodyAdvice 中检查请求路径

你可以在 beforeBodyWrite 方法中通过 ServerHttpRequest 对象获取请求的 URL,然后检查是否匹配特定的路径。这样可以确保只有匹配的请求路径被拦截并处理。

示例如下:

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;

@Component
public class CommonResultResponseBodyAdvice implements ResponseBodyAdvice<Object> {

    private final ObjectMapper objectMapper = new ObjectMapper();

    @Override
    public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
        // 仅处理返回类型为 CommonResult 的响应
        return CommonResult.class.isAssignableFrom(returnType.getParameterType());
    }

    @Override
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType,
                                  Class<? extends HttpMessageConverter<?>> selectedConverterType,
                                  ServerHttpRequest request, ServerHttpResponse response) {
        // 获取请求的 URI
        String path = request.getURI().getPath();

        // 判断请求路径是否以 /zjdl/huz/performance/board 开头
        if (path.startsWith("/zjdl/huz/performance/board")) {
            if (body instanceof CommonResult) {
                CommonResult<?> result = (CommonResult<?>) body;

                // 使用 Jackson 将 data 转换为 JsonNode 处理
                JsonNode dataNode = objectMapper.valueToTree(result.getData());
                StringUtil.processJsonNode(dataNode);

                // 将处理后的 JsonNode 重新设置回 CommonResult
                result.setData(objectMapper.convertValue(dataNode, new TypeReference<Object>() {}));
            }
        }

        return body;
    }
}

2. 在拦截器中控制 ResponseBodyAdvice

另一种方式是结合 HandlerInterceptorResponseBodyAdvice,在拦截器中设置一个标记,指示当前请求是否需要由 ResponseBodyAdvice 处理。

2.1 自定义拦截器
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;

@Component
public class CustomInterceptor implements HandlerInterceptor {

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        // 在请求处理之前执行
        if (request.getRequestURI().startsWith("/zjdl/huz/performance/board")) {
            request.setAttribute("intercepted", true);
        }
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        // 请求处理之后但在视图渲染之前执行
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        // 请求完成后执行
    }
}
2.2 修改 ResponseBodyAdvice 判断标记

ResponseBodyAdvice 中,你可以检查请求中是否包含这个标记来决定是否进行处理。

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.core.MethodParameter;
import org.springframework.http.MediaType;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice;

@Component
public class CommonResultResponseBodyAdvice implements ResponseBodyAdvice<Object> {

    private final ObjectMapper objectMapper = new ObjectMapper();

    @Override
    public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
        // 仅处理返回类型为 CommonResult 的响应
        return CommonResult.class.isAssignableFrom(returnType.getParameterType());
    }

    @Override
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType,
                                  Class<? extends HttpMessageConverter<?>> selectedConverterType,
                                  ServerHttpRequest request, ServerHttpResponse response) {
        // 通过 HttpServletRequest 获取标记
        HttpServletRequest servletRequest = (HttpServletRequest) request;
        Boolean intercepted = (Boolean) servletRequest.getAttribute("intercepted");

        if (Boolean.TRUE.equals(intercepted)) {
            if (body instanceof CommonResult) {
                CommonResult<?> result = (CommonResult<?>) body;

                // 使用 Jackson 将 data 转换为 JsonNode 处理
                JsonNode dataNode = objectMapper.valueToTree(result.getData());
                StringUtil.processJsonNode(dataNode);

                // 将处理后的 JsonNode 重新设置回 CommonResult
                result.setData(objectMapper.convertValue(dataNode, new TypeReference<Object>() {}));
            }
        }

        return body;
    }
}

3. 配置拦截器

最后,将拦截器配置到 Spring Boot 项目中。

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration
public class WebConfig implements WebMvcConfigurer {

    @Autowired
    private CustomInterceptor customInterceptor;

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(customInterceptor)
                .addPathPatterns("/zjdl/huz/performance/board/**");
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

舒一笑不秃头

你的鼓励是我创作的最大动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值