Что под капотом Spring Data JPA? | OTUS
⚡Подписка от OTUS!
Собери свой пул курсов на выгодных условиях. Подробности в чате →
Написать в чат

Курсы

Программирование
Unity Game Developer. Basic
-15%
Разработчик программных роботов (RPA) на базе UiPath и PIX
-6%
Разработчик C#
-8%
Алгоритмы и структуры данных
-8%
Backend-разработчик на PHP
-8%
JavaScript Developer. Professional
-9%
iOS Developer. Professional
-8%
Базы данных
-12%
C# ASP.NET Core разработчик
-6%
Python Developer. Basic
-10%
Java Developer. Professional Web-разработчик на Python Android Developer. Basic PostgreSQL Software Architect Reverse-Engineering. Professional Kotlin Backend Developer React.js Developer VOIP инженер Нереляционные базы данных Scala-разработчик Супер-практикум по использованию и настройке GIT IoT-разработчик JavaScript Developer. Basic Advanced Fullstack JavaScript developer Unity Game Developer. Professional Супер-интенсив Azure
Инфраструктура
Супер-интенсив "Версионирование и командная работа с помощью Git"
-30%
Administrator Linux. Professional
-5%
Супер-интенсив «CI/CD или Непрерывная поставка с Docker и Kubernetes»
-30%
Разработчик программных роботов (RPA) на базе UiPath и PIX
-6%
Administrator Linux. Advanced
-8%
Infrastructure as a code in Ansible
-12%
Network engineer
-4%
MS SQL Server Developer
-8%
Cloud Solution Architecture Highload Architect Разработчик голосовых ассистентов и чат-ботов Мониторинг и логирование: Zabbix, Prometheus, ELK Супер-практикум по работе с протоколом BGP Супер - интенсив по паттернам проектирования Супер - интенсив по Kubernetes Архитектор сетей Супер-интенсив «IaC Ansible»
Специализации Курсы в разработке Подготовительные курсы
+7 499 938-92-02

Что под капотом Spring Data JPA?

JavaDeep_22.06_Site.png

Тоже впечатлены элегантностью работы с Retrofit и Spring Data JPA? Когда вы просто добавляете в интерфейс сигнатуру метода, а при вызове этого метода оказывается, что он реально работает!

Да, долой boilerplate* code!

Так это выглядит при работе с Spring Data JPA. И никаких SQL, HQL или JPQL!

interface PersonRepository extends Repository<Person, Long> {
  List<Person> findByLastname(String lastname);
  Long countByLastname(String lastname);
}

Знаете что там внутри? Ничего нового – обычный Java Reflection API!

Когда библиотека получает ваш интерфейс, ей нужно решить два главных вопроса: 1. Как в процессе выполнения программы создать экземпляр класса, реализующий ваш интерфейс? Ведь библиотека должна дать вам возможность вызова этих методов так, как будто это методы самого обычного класса. 2. Как получить и проанализировать сигнатуры методов, которые вы описали в интерфейсе? Ведь на основании соглашений о наименовании метода интерфейса библиотеке необходимо обработать вызов этого метода определённым способом.

Класс java.lang.reflect.Proxy создан как раз для этого. Он позволяет создать динамический прокси-объект, реализующий указанный интерфейс.

В таком прокси-объекте все вызовы его методов перехватываются и направляются в специальный обработчик – реализацию интерфейса InvocationHandler.

Вот как это выглядит:

PersonRepository personRepository = (PersonRepository) Proxy.newProxyInstance(PersonRepository.class.getClassLoader(),
                new Class[] { PersonRepository.class },
                (proxy, method, args) -> {
                    String methodName = method.getName();
                    // Обрабатываем вызов метода интерфейса, возвращаем результат
                    if(methodName.startsWith("find")) {
                        // ...
                    }

                    return new Object();
                });

Лямбда-выражение (proxy, method, args) реализует интерфейс InvocationHandler и содержит логику обработки вызовов всех методов PersonRepository.

InvocationHandler содержит все необходимые для обработки вызова метаданные: – ссылку на объект, – метод которого вызвал пользователь, – сигнатуру вызванного метода с его названием, параметрами и аннотациями, – и, конечно, значения аргументов, переданных пользователем при вызове метода.

В использовании полученный прокси-объект не отличается от обыкновенного класса:

List<Person> persons = personRepository.findByLastname("Otus");

Есть вопрос? Напишите в комментариях! *Boilerplate_code

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

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

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

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