Spring FrameworkMiddleExperience

Какие архитектурные решения Spring Framework задаёт вокруг request lifecycle, middleware, dependency model, validation и error handling?

DispatcherServlet управляет lifecycle через HandlerMapping, HandlerInterceptor и MessageConverter; middleware реализуется фильтрами и интерсепторами; DI через IoC-контейнер; валидация через Bean Validation API; ошибки — через @RestControllerAdvice.

Request Lifecycle в Spring MVC

Каждый HTTP-запрос проходит следующий путь: DispatcherServlet принимает запрос, делегирует его HandlerMapping для поиска нужного контроллера, затем вызывает цепочку HandlerInterceptor (preHandle), далее — сам метод контроллера через HandlerAdapter, после — postHandle, затем ViewResolver или HttpMessageConverter для сериализации ответа, и наконец afterCompletion.

@Component
public class LoggingInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest req,
                             HttpServletResponse res,
                             Object handler) {
        System.out.println("Request: " + req.getMethod() + " " + req.getRequestURI());
        return true; // false — прерывает цепочку
    }

    @Override
    public void afterCompletion(HttpServletRequest req,
                                HttpServletResponse res,
                                Object handler,
                                Exception ex) {
        System.out.println("Completed");
    }
}

@Configuration
public class WebConfig implements WebMvcConfigurer {
    @Autowired LoggingInterceptor loggingInterceptor;

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(loggingInterceptor)
                .addPathPatterns("/api/**");
    }
}

Middleware — Servlet Filters и HandlerInterceptors

Spring предоставляет два уровня «middleware»:

  • Servlet Filter — работает на уровне сервлет-контейнера, до DispatcherServlet. Подходит для CORS, аутентификации, логирования raw-запросов. Регистрируется через FilterRegistrationBean или аннотацию @Component + реализация Filter.
  • HandlerInterceptor — внутри Spring MVC, имеет доступ к HandlerMethod. Лучше подходит для авторизации по ролям, A/B-тестирования, tenant-разрешений.
@Component
public class JwtFilter extends OncePerRequestFilter {
    @Override
    protected void doFilterInternal(HttpServletRequest req,
                                    HttpServletResponse res,
                                    FilterChain chain)
            throws ServletException, IOException {
        String token = req.getHeader("Authorization");
        // validate token...
        chain.doFilter(req, res);
    }
}

Dependency Model — IoC Container

Spring ApplicationContext управляет бинами. Область видимости по умолчанию — singleton (один экземпляр на контекст). Доступные скоупы: prototype, request, session, application. Инъекция зависимостей — через конструктор (рекомендуется), поле (@Autowired) или сеттер. Бины создаются при старте контекста и проходят цикл: BeanDefinition → instantiation → populateProperties → @PostConstruct / InitializingBean.afterPropertiesSet() → ready.

@Service
public class OrderService {
    private final PaymentGateway gateway;
    private final OrderRepository repo;

    // Constructor injection — immutable, testable
    public OrderService(PaymentGateway gateway, OrderRepository repo) {
        this.gateway = gateway;
        this.repo = repo;
    }
}

Validation

Spring интегрируется с Bean Validation (Jakarta Validation API). Аннотации @Valid / @Validated на параметрах метода контроллера запускают валидацию через MethodValidationInterceptor. Ошибки перехватываются через BindingResult или глобальным @ExceptionHandler(MethodArgumentNotValidException.class).

public record CreateUserRequest(
    @NotBlank String username,
    @Email String email,
    @Min(18) int age
) {}

@RestController
@RequestMapping("/users")
public class UserController {
    @PostMapping
    public ResponseEntity<User> create(@Valid @RequestBody CreateUserRequest req) {
        // ...
    }
}

@RestControllerAdvice
public class GlobalExceptionHandler {
    @ExceptionHandler(MethodArgumentNotValidException.class)
    public ResponseEntity<Map<String, String>> handleValidation(
            MethodArgumentNotValidException ex) {
        Map<String, String> errors = new HashMap<>();
        ex.getBindingResult().getFieldErrors()
          .forEach(e -> errors.put(e.getField(), e.getDefaultMessage()));
        return ResponseEntity.badRequest().body(errors);
    }
}

Error Handling

Централизованная обработка ошибок реализуется через @RestControllerAdvice + @ExceptionHandler. Альтернатива — ResponseEntityExceptionHandler как базовый класс с готовыми обработчиками стандартных Spring-исключений. Для кастомных HTTP-статусов используется @ResponseStatus на классе исключения.

@ResponseStatus(HttpStatus.NOT_FOUND)
public class ResourceNotFoundException extends RuntimeException {
    public ResourceNotFoundException(String id) {
        super("Resource not found: " + id);
    }
}

Подводные камни

  • Порядок фильтров важен: если SecurityFilterChain стоит после вашего фильтра, токен не будет распознан при первом запросе.
  • @Transactional не работает при self-invocation — вызов транзакционного метода из того же бина обходит AOP-прокси.
  • Scope mismatch: инъекция request-скоупного бина в singleton требует @Scope(proxyMode = ScopedProxyMode.TARGET_CLASS).
  • HandlerInterceptor не вызывается, если фильтр прервал цепочку до DispatcherServlet.
  • @Valid не работает без зависимости spring-boot-starter-validation — Spring Boot не включает её по умолчанию с версии 2.3.
  • Глобальный @ExceptionHandler в @ControllerAdvice не перехватывает ошибки фильтров — для этого нужен отдельный Filter с try/catch.
  • Бины с prototype-скоупом не уничтожаются контейнером — @PreDestroy не вызывается, утечка ресурсов.
  • Неправильная регистрация ControllerAdvice: если он находится в пакете вне @ComponentScan, Spring его не найдёт.

What hurts your answer

  • Знать термины Spring Framework, но не понимать связи между абстракциями
  • Объяснять поведение через отдельные примеры вместо причинной модели
  • Не связывать mental model с диагностикой ошибок

What they're listening for

  • Понимает ключевые абстракции Spring Framework
  • Может предсказывать поведение системы через mental model
  • Связывает модель с debugging и production decisions

Related topics