Java过滤器-拦截器-AOP-Controller

1 对比

组件 核心用途 主要应用场景
‌**过滤器 (Filter)**‌ 在请求到达Servlet之前或响应返回客户端之前,对原始的ServletRequestServletResponse进行预处理或后处理。 字符编码设置、敏感词过滤、URL级别的权限认证、请求日志记录。‌12
‌**拦截器 (Interceptor)**‌ 在请求进入Controller方法前后,以及视图渲染前后,对请求进行拦截和处理。它基于Spring的反射机制。 权限认证、审计日志、请求参数预处理、响应数据包装。‌12
‌**切面 (AOP)**‌ 通过"切面"技术,在不修改目标方法代码的情况下,横向切入业务逻辑,实现功能的扩展和解耦。它基于动态代理(如JDK Proxy或CGLIB)。 方法级别的权限控制、日志记录、事务管理、缓存操作、方法执行时间监控。‌13
ControllerAdvice 全局异常处理和数据绑定/预处理,专门用于增强Controller层。(也是基于AOP) 统一的全局异常处理、统一的数据格式化、统一的请求参数预处理。‌13

使用过程中需要注意执行顺序和和优先级。

2 Servlet

在Java Web应用程序中,Servlet是处理HTTP请求和响应的核心组件,Servlet的生命周期:

1 加载和实例化

当服务器启动时,会加载web.xml中配置的Servlet或者在注解为@WebServlet的类。

服务器将为每个Servlet创建一个实例。

  1. 初始化

在Servlet实例创建后,会调用init(ServletConfig config)方法进行初始化。

这个方法通常用于执行仅需执行一次的初始化操作,比如加载资源文件、打开数据库连接等。

  1. 请求处理

当客户端发送请求到服务器时,服务器会调用Servlet的service(ServletRequest req, ServletResponse res)方法。

在Servlet 3.0及更高版本中,推荐使用doGet(HttpServletRequest request, HttpServletResponse response)

和doPost(HttpServletRequest request, HttpServletResponse response)等方法来分别处理GET和POST请求。

  1. 服务

在每次接收到新的请求时,service方法或相应的doXxx方法会被调用,用于处理请求并生成响应。

  1. 销毁

当Web应用被卸载或服务器关闭时,会调用Servlet的destroy()方法。这个方法用于释放资源,比如关闭数据库连接、停止线程等。

注意事项:

‌单例性‌:默认情况下,Servlet是单例的,即整个Web应用生命周期内只会有一个Servlet实例。这意味着所有的客户端请求将共享同一个Servlet实例。如果Servlet需要处理大量并发请求,应确保其线程安全。

‌多线程‌:由于Servlet是单例的,多个请求可能会并发地执行service方法或其子方法(如doGet, doPost)。因此,编写Servlet时需要考虑线程安全。可以使用同步块或并发工具类来管理共享资源的访问。

‌生命周期管理‌:可以通过覆盖init和destroy方法来管理资源的初始化和清理工作。对于非单例的Servlet,可以使用@WebServlet(value="/path", loadOnStartup=1)注解来控制加载顺序和时机

3 动态代理

维度 Java动态代理‌(基于JDK) CGLIB代理‌(基于ASM)
实现原理 通过反射机制生成实现目标接口的代理类 通过ASM字节码操作生成目标类的子类
适用场景 目标类必须实现接口 目标类无需实现接口,支持类级别的代理
性能 通常稍逊于CGLIB,但差异可忽略 性能更优,尤其在无接口场景下
限制 无法代理final类或方法 无法代理final类
实现方式 java.lang.reflect.Proxy + InvocationHandler net.sf.cglib.proxy.Enhancer
使用场景 接口多、结构清晰的项目 无接口类、老项目或性能敏感场景

JDK Proxy

反射和接口

java 复制代码
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

// 定义目标接口
interface TargetInterface {
    void doSomething();
}

// 实现目标接口
class TargetImpl implements TargetInterface {
    @Override
    public void doSomething() {
        System.out.println("TargetImpl method");
    }
}

// 代理处理器
class TargetProxy implements InvocationHandler {
    private TargetInterface targetInterface;

    public TargetProxy(TargetInterface targetInterface) {
        this.targetInterface = targetInterface;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println(method.getName() + "前置处理");
        Object result = method.invoke(targetInterface, args);
        System.out.println(method.getName() + "后置处理");
        return result;
    }

    public TargetInterface getProxyInstance() {
        return (TargetInterface) Proxy.newProxyInstance(
            TargetProxy.class.getClassLoader(),
            targetInterface.getClass().getInterfaces(),
            this
        );
    }
}

// 测试类
public class JDKDynamicProxyExample {
    public static void main(String[] args) {
        TargetInterface target = new TargetImpl();
        TargetProxy proxyHandler = new TargetProxy(target);
        TargetInterface proxy = proxyHandler.getProxyInstance();
        proxy.doSomething();
    }
}

CGLIb

SM字节码操作生成目标类的子类

java 复制代码
public class TargetClass {
    public void doSomething() {
        System.out.println("TargetClass.doSomething()执行");
    }
    
    public String getValue(String input) {
        return "处理结果: " + input;
    }
}

import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;

public class MyMethodInterceptor implements MethodInterceptor {
    @Override
    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("前置增强: " + method.getName());
        Object result = proxy.invokeSuper(obj, args); // 调用父类方法
        System.out.println("后置增强: " + method.getName());
        return result;
    }
}

import net.sf.cglib.proxy.Enhancer;

public class CglibProxyDemo {
    public static void main(String[] args) {
        // 创建Enhancer对象
        Enhancer enhancer = new Enhancer();
        // 设置父类
        enhancer.setSuperclass(TargetClass.class);
        // 设置回调
        enhancer.setCallback(new MyMethodInterceptor());
        // 创建代理对象
        TargetClass proxy = (TargetClass) enhancer.create();
        
        // 调用方法
        proxy.doSomething();
        String result = proxy.getValue("CGLIB测试");
        System.out.println(result);
    }
}
相关推荐
guslegend2 小时前
Spring AOP高级应用与源码剖析
java
Rover.x2 小时前
head table is mandatory
java·apache
shoubepatien2 小时前
JAVA —— 03
java·jvm
a努力。2 小时前
【基础数据篇】数据等价裁判:Comparer模式
java·后端
小冷coding2 小时前
【Java】高并发架构设计:1000 QPS服务器配置与压测实战
java·服务器·开发语言
哈哈哈笑什么2 小时前
SpringBoot 企业级接口加密【通用、可配置、解耦的组件】「开闭原则+模板方法+拦截器/中间件模式」
java·后端·安全
期待のcode2 小时前
springboot依赖管理机制
java·spring boot·后端
破刺不会编程2 小时前
socket编程TCP
linux·运维·服务器·开发语言·网络·网络协议·tcp/ip
北极糊的狐2 小时前
IDEA中安装 CamelCase 插件实现变量快速生成驼峰命名
开发语言·php