CDI Interceptors | OTUS
⚡ Подписка на курсы OTUS!
Интенсивная прокачка навыков для IT-специалистов!
Подробнее

Курсы

Программирование
iOS Developer. Professional Kotlin Backend Developer Flutter Mobile Developer Symfony Framework C++ Developer. Basic Unity Game Developer. Basic Java Developer. Professional
-35%
Highload Architect Unity Game Developer. Professional React.js Developer Специализация Java-разработчик
-25%
Алгоритмы и структуры данных
-16%
Scala-разработчик C# Developer. Professional
-23%
Разработчик голосовых ассистентов и чат-ботов Team Lead Архитектура и шаблоны проектирования NoSQL Web-разработчик на Python Golang Developer. Professional PostgreSQL Vue.js разработчик Супер-практикум по использованию и настройке GIT Разработчик IoT Подготовка к сертификации Oracle Java Programmer (OCAJP) Программист С HTML/CSS
Инфраструктура
Инфраструктурная платформа на основе Kubernetes Microservice Architecture Базы данных Highload Architect Reverse-Engineering. Professional
-8%
Network engineer. Basic Administrator Linux.Basic MongoDB Infrastructure as a code MS SQL Server Developer Cloud Solution Architecture Мониторинг и логирование: Zabbix, Prometheus, ELK Супер-практикум по использованию и настройке GIT Разработчик IoT Экcпресс-курс «ELK» Супер-интенсив "Tarantool" Экспресс-курс «CI/CD или Непрерывная поставка с Docker и Kubernetes» Экспресс-курс «Введение в непрерывную поставку на базе Docker»
Корпоративные курсы
Безопасность веб-приложений Экосистема Hadoop, Spark, Hive Пентест. Практика тестирования на проникновение Node.js Developer Java QA Engineer. Basic
-18%
Reverse-Engineering. Professional
-8%
DevOps практики и инструменты NoSQL Reverse-Engineering. Basic Cloud Solution Architecture Внедрение и работа в DevSecOps Супер-практикум по работе с протоколом BGP Game QA Engineer Супер - интенсив по Kubernetes Дизайн сетей ЦОД Экспресс-курс «IaC Ansible» Экспресс-курс по управлению миграциями (DBVC) Экспресс-курс "Версионирование и командная работа с помощью Git" Основы Windows Server
Специализации Курсы в разработке Подготовительные курсы Подписка
+7 499 938-92-02

CDI Interceptors

JavaEE_Deep_12-5020-1ade4a.09_site.png

В спецификации CDI предусмотрен механизм «перехватчиков», который позволяет реализовывать в прикладном коде приёмы аспектно-ориентированного программирования. Иначе говоря, разработчику становится доступна сквозная функциональность (cross cutting concerns), то есть прежде, чем обратиться к некоторому методу CDI-бина, у программиста появляется возможность выполнить предобработку, сам бизнес-метод и постобработку.

Для этих целей и существует механизм CDI Interceptors, причём применяется он не только для спецификации CDI, но и активно используется в технологии Enterprise JavaBeans.

Для связи бизнес-метода с кодом аспектно-ориентированного программирования необходимо создать кастомную аннотацию, которая дополнительно помечается аннотацией @InterceptorBinding:

@InterceptorBinding
@Target( { METHOD, TYPE } )
@Retention( RUNTIME )
public @interface Audited {
}

Запуск работы Interceptors включает создание класса AuditedInterceptor, над которым следует разместить две аннотаци: @Interceptor (из коробки CDI) и только что созданную пользовательскую аннотацию @Audited.

Ключевым методом соответствующего класса перехватчика является метод, помеченный аннотацией @AroundInvoke. Для наглядности переменные calledBefore и calledAfter определяют места для пред- и постобработки, поэтому в приведённом ниже примере они выставляются в истинное значение до и после выполнения метода proceed().

Вызов этого метода у объекта InvocationContext позволяет выполнять код бизнес-метода. Без данной конструкции передача управления в метод бина не будет осуществляться, что в определённых случаях позволяет менять привычный порядок выполнения приложения (например, фильтр проверки ролей пользователя).

@Audited
@Interceptor
public class AuditedInterceptor {
    public static boolean calledBefore = false;
    public static boolean calledAfter = false;

    @AroundInvoke
    public Object auditMethod(InvocationContext ctx) throws Exception {
        calledBefore = true;
        Object result = ctx.proceed();
        calledAfter = true;
        return result;
    }
}

Теперь осталось пометить бизнес-метод самого CDI-бина аннотацией @Audited, которую мы уже неоднократно упоминали:

public class SuperService {
    @Audited
    public String deliverService(String uid) {
        return uid;
    }
}

Если аннотация помечается над классом бина, то механизм перехватчиков задействуется для всех его методов.

Кроме того, чтобы интерцептор заработал в приложении, для спецификации CDI важно определить его в конфигурационном файле beans.xml:

<beans xmlns="http://java.sun.com/xml/ns/javaee"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
      http://java.sun.com/xml/ns/javaee/beans_1_2.xsd">
    <interceptors>
        <class>ru.otus.interceptor.AuditedInterceptor</class>
    </interceptors>
</beans>

Осталось рассмотреть преимущества и недостатки CDI Interceptors

Начнём с плюсов: - использование CDI Interceptors является стандартной фичей спецификации Java EE; - CDI-библиотеки могут применяться в приложениях Java SE; - CDI Interceptors могут использоваться, когда есть серьёзные ограничения на сторонние библиотеки в рамках приложений Java EE.

А теперь минусы: - возникает жёсткая связка между классом бизнес-логики и классом интерцептора; - трудно определить, какие классы подвергаются интерцепции в проекте; - нет гибкого механизма применения CDI Interceptors к некоторой группе методов, т. е. нужно над каждым вешать соответствующую аннотацию.

Пожалуй, на этом всё. Узнать больше можно на курсах OTUS по Java EE.

Возникли сложности при использовании CDI Interceptors? Задавайте вопросы в комментариях!

Не пропустите новые полезные статьи!

Спасибо за подписку!

Мы отправили вам письмо для подтверждения вашего email.
С уважением, OTUS!

Автор
0 комментариев
Для комментирования необходимо авторизоваться