springboot 中拦截器与过滤器的区别是什么,两种方式的使用场景分别是什么

springboot 中拦截器与过滤器的区别是什么,两种方式的使用场景分别是什么?

拦截器(Interceptor)和过滤器(Filter)是Spring Boot中常用的两种请求处理机制,它们有以下区别和使用场景:

区别:

  1. 拦截器是基于Java反射机制实现的,而过滤器是基于Servlet规范实现的。
  2. 拦截器只能对Spring MVC的请求进行拦截处理,而过滤器可以对所有的请求进行拦截处理,包括静态资源请求。
  3. 拦截器可以访问Spring MVC的上下文(如HandlerMethod等),而过滤器无法直接访问。

使用场景:

  1. 拦截器适用于对特定的Controller请求进行预处理、后处理、日志记录等操作,如权限验证、日志记录等。
  2. 过滤器适用于对所有的请求进行统一的处理,如字符编码转换、请求参数过滤、请求响应压缩等。

总结:

  • 如果需要对特定的Controller请求进行处理,可以使用拦截器。
  • 如果需要对所有的请求进行统一处理,包括静态资源请求,可以使用过滤器。

想要对 post 请求中的 request body 参数进行统一解密,然后再继续调用接口,这个是使用拦截器还是过滤器?

对于对 POST 请求中的请求体参数进行解密的需求,可以使用过滤器来实现。

过滤器可以在请求到达控制器之前拦截请求,并对请求进行处理。在过滤器中,您可以获取请求的原始数据,包括请求体参数,并对其进行解密操作。然后,您可以将解密后的参数设置回请求中,再继续传递给下一个处理器(如控制器)。

使用过滤器的好处是,它可以对所有的请求进行统一处理,而不仅限于特定的控制器。这样可以确保所有的 POST 请求都经过解密处理,提高代码的可维护性和复用性。

因此,建议使用过滤器来对 POST 请求中的请求体参数进行解密操作。

springboot 过滤器 AES 解密示例

在 Spring Boot 中,可以通过实现 javax.servlet.Filter 接口来创建过滤器,并将其注册到应用程序中。下面是一个使用过滤器实现 AES 解密逻辑的示例代码:

import org.springframework.stereotype.Component;
import org.springframework.web.filter.OncePerRequestFilter;
import javax.crypto.Cipher;
import javax.crypto.spec.SecretKeySpec;
import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.Base64;

@Component
public class AesDecryptionFilter extends OncePerRequestFilter {

    private static final String SECRET_KEY = "mysecretkey12345";

    @Override
    protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
        String encryptedValue = request.getParameter("data");
        if (encryptedValue != null) {
            try {
                String decryptedValue = decrypt(encryptedValue, SECRET_KEY);
                request.setAttribute("decryptedData", decryptedValue);
            } catch (Exception e) {
                // 解密失败,可以根据实际情况进行处理
            }
        }
        filterChain.doFilter(request, response);
    }

    private String decrypt(String encryptedValue, String secretKey) throws Exception {
        byte[] decodedKey = Base64.getDecoder().decode(secretKey.getBytes(StandardCharsets.UTF_8));
        SecretKeySpec keySpec = new SecretKeySpec(decodedKey, "AES");
        Cipher cipher = Cipher.getInstance("AES/ECB/PKCS5Padding");
        cipher.init(Cipher.DECRYPT_MODE, keySpec);
        byte[] decodedValue = Base64.getDecoder().decode(encryptedValue.getBytes(StandardCharsets.UTF_8));
        byte[] decryptedValue = cipher.doFinal(decodedValue);
        return new String(decryptedValue, StandardCharsets.UTF_8);
    }
}

上面的示例中,AesDecryptionFilter 类继承了 OncePerRequestFilter 类,这意味着过滤器只会在每个请求中执行一次。在 doFilterInternal 方法中,我们首先检查请求中是否包含名为 data 的参数,如果存在,则进行解密并将解密后的值设置为请求属性 decryptedData。最后,我们使用 filterChain.doFilter(request, response) 调用过滤器链中的下一个过滤器。

在解密方法中,我们使用 Base64 解码密钥和加密数据,并使用 Cipher 类执行 AES 解密操作。请注意,此示例假定使用的是 AES 的 ECB 模式和 PKCS5 填充。在实际使用中,可能需要根据具体情况进行调整,例如选择不同的模式(如 CBC)和填充方式。

最后,在应用程序中注册过滤器。可以通过在 Spring Boot 应用程序的配置类中添加以下代码来完成此操作:

@Configuration
public class AppConfig {

    @Autowired
    private AesDecryptionFilter aesDecryptionFilter;

    @Bean
    public FilterRegistrationBean<AesDecryptionFilter> loggingFilter() {
        FilterRegistrationBean<AesDecryptionFilter> registrationBean = new FilterRegistrationBean<>();
        registrationBean.setFilter(aesDecryptionFilter);
        registrationBean.addUrlPatterns("/api/*"); // 指定要应用过滤器的 URL 模式
        return registrationBean;
    }
}

上面的示例中,我们创建了一个 FilterRegistrationBean 对象,并将其设置为 AesDecryptionFilter 的实例。然后,我们使用 addUrlPatterns 方法指定要应用过滤器的 URL 模式。在这个例子中,我们假设所有 API 请求都以 /api/ 开头。

springboot 过滤器对 post body 参数解密示例


@Component
public class AesDecryptionFilter extends OncePerRequestFilter {

    private static final String SECRET_KEY =  "mfsecretkey14523";
    private static final String INIT_VECTOR = "mfinitialve1ctor";

    private static final String DATA_STR = "data";
    private static final String ENCRYPT_DATA_STR = "encryptedData";

    /**
     * 可以作为类成员变量重复使用
     **/
    private ObjectMapper objectMapper = new ObjectMapper();

    @Override
    protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {

        // 检查是否是POST请求并且内容类型是JSON
        if ("POST".equalsIgnoreCase(request.getMethod()) && "application/json".equalsIgnoreCase(request.getContentType())) {
            // 包装HttpServletRequest以便可以多次读取InputStream
            HttpServletRequestWrapper wrappedRequest = new HttpServletRequestWrapper(request) {
                private byte[] rawData;
                private boolean isDecrypted = false;

                @Override
                public ServletInputStream getInputStream() throws IOException {
                    if (rawData == null) {
                        rawData = IOUtils.toByteArray(super.getInputStream());
                        if (!isDecrypted) {
                            // 解密操作
                            try {
                                String requestBodyStr = new String(rawData, StandardCharsets.UTF_8);
                                JsonNode rootNode = objectMapper.readTree(requestBodyStr);
                                JsonNode encryptedNode = rootNode.get(ENCRYPT_DATA_STR);
                                if (encryptedNode != null) {
                                    String encryptedData = encryptedNode.asText();
                                    String decryptedValue = decrypt(encryptedData);
                                    rawData = decryptedValue.getBytes(StandardCharsets.UTF_8);
                                    isDecrypted = true;
                                }
                            } catch (Exception e) {
                                // 解密失败,可以根据实际情况进行处理
                                logger.error("doFilterInternal decrypt error: ", e);
                                throw new IOException("Error decrypting the request body", e);
                            }
                        }
                    }
                    return new ServletInputStreamWrapper(rawData);
                }

                @Override
                public BufferedReader getReader() throws IOException {
                    return new BufferedReader(new InputStreamReader(this.getInputStream(), StandardCharsets.UTF_8));
                }
            };
            filterChain.doFilter(wrappedRequest, response);
        } else {
            // 对于非POST或非JSON请求,不做处理直接传递给下一个过滤器
            filterChain.doFilter(request, response);
        }
    }

    private String decrypt(String encrypt_data) throws Exception {
        Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5PADDING");
        SecretKeySpec key = new SecretKeySpec(SECRET_KEY.getBytes(StandardCharsets.UTF_8), "AES");
        IvParameterSpec iv = new IvParameterSpec(INIT_VECTOR.getBytes(StandardCharsets.UTF_8));
        cipher.init(Cipher.DECRYPT_MODE, key, iv);
        byte[] original = cipher.doFinal(Base64.getDecoder().decode(encrypt_data));
        return new String(original);
    }


    private class ServletInputStreamWrapper extends ServletInputStream {
        private final ByteArrayInputStream bais;

        public ServletInputStreamWrapper(byte[] rawData) {
            this.bais = new ByteArrayInputStream(rawData);
        }

        @Override
        public int read() throws IOException {
            return bais.read();
        }

        @Override
        public boolean isFinished() {
            return bais.available() == 0;
        }

        @Override
        public boolean isReady() {
            return true;
        }

        @Override
        public void setReadListener(ReadListener readListener) {
            throw new RuntimeException("Not implemented");
        }
    }
}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容