🧩 Микросервисы: Укрощение хаоса (Spring Cloud)
Когда у вас один сервис, всё просто. Но когда их становится 10, 20 или 50, возникают вопросы:
1. Как сервису А узнать IP-адрес сервиса Б? (А если он меняется динамически?)
2. Как не писать тонны кода для HTTP-запросов?
3. Как клиенту (фронтенду) обращаться ко всей этой куче сервисов?
Для этого есть три главных инструмента.
1️⃣ Eureka: Телефонная книга (Service Discovery)
В облаке сервисы постоянно перезапускаются, меняют IP-адреса и порты. Хардкодить
Eureka Server - это реестр.
🔴 Когда сервис (например,
🔴 Когда
В коде:
Вам нужно просто добавить аннотацию
2️⃣ OpenFeign: Магия общения
Окей, адрес мы нашли. Теперь нужно отправить запрос.
Раньше мы использовали
Feign позволяет вызывать удаленный REST-сервис так, будто это обычный метод интерфейса в вашем коде.
Было (
Стало (
Вы просто пишете интерфейс, а реализацию Spring сгенерирует сам!
Это называется Декларативный REST-клиент. Чисто, красиво, типизировано.
3️⃣ API Gateway: Единая точка входа
Представьте, что у вас 50 микросервисов. Фронтенд не должен знать адреса каждого из них (
Spring Cloud Gateway - это вахтер на входе.
Весь внешний мир стучится только в него (например, на порт 8080), а он уже сам разруливает, куда отправить запрос.
Что он делает:
1. Маршрутизация:
2. Безопасность: Проверяет JWT токен один раз на входе.
3. Rate Limiting: "Не больше 10 запросов в секунду от этого юзера".
Пример конфига (
⚡ Итог: Как это работает вместе?
1. Сервисы просыпаются и регистрируются в Eureka.
2. Фронтенд шлет запрос на Gateway.
3. Gateway спрашивает у Eureka адрес нужного сервиса и пересылает запрос.
4. Если сервисам нужно пообщаться между собой, они используют Feign.
#Java #Microservices #SpringCloud #Eureka #Feign
📲 Мы в MAX
👉@BookJava
Когда у вас один сервис, всё просто. Но когда их становится 10, 20 или 50, возникают вопросы:
1. Как сервису А узнать IP-адрес сервиса Б? (А если он меняется динамически?)
2. Как не писать тонны кода для HTTP-запросов?
3. Как клиенту (фронтенду) обращаться ко всей этой куче сервисов?
Для этого есть три главных инструмента.
1️⃣ Eureka: Телефонная книга (Service Discovery)
В облаке сервисы постоянно перезапускаются, меняют IP-адреса и порты. Хардкодить
https://localhost:8081 в коде - самоубийство.Eureka Server - это реестр.
OrderService) запускается, он звонит в Eureka: "Привет, я OrderService, мой IP такой-то".UserService хочет найти заказы, он спрашивает Eureka: "Где сейчас живет OrderService?".В коде:
Вам нужно просто добавить аннотацию
@EnableDiscoveryClient, и магия произойдет сама. Сервисы будут находить друг друга по имени, а не по IP.2️⃣ OpenFeign: Магия общения
Окей, адрес мы нашли. Теперь нужно отправить запрос.
Раньше мы использовали
RestTemplate - это было громоздко и некрасиво.Feign позволяет вызывать удаленный REST-сервис так, будто это обычный метод интерфейса в вашем коде.
Было (
RestTemplate):
String url = "https://order-service/orders/" + userId;
List<Order> orders = restTemplate.getForObject(url, List.class); // Фу, нетипизированно
Стало (
FeignClient):Вы просто пишете интерфейс, а реализацию Spring сгенерирует сам!
@FeignClient(name = "order-service") // Имя сервиса в Eureka
public interface OrderClient {
@GetMapping("/orders/{userId}")
List<Order> getUserOrders(@PathVariable Long userId);
}
// Использование в сервисе:
// List<Order> orders = orderClient.getUserOrders(123L);
Это называется Декларативный REST-клиент. Чисто, красиво, типизировано.
3️⃣ API Gateway: Единая точка входа
Представьте, что у вас 50 микросервисов. Фронтенд не должен знать адреса каждого из них (
api.com:8081, api.com:8082...). Это небезопасно и сложно (CORS error, привет 👋).Spring Cloud Gateway - это вахтер на входе.
Весь внешний мир стучится только в него (например, на порт 8080), а он уже сам разруливает, куда отправить запрос.
Что он делает:
1. Маршрутизация:
/users/** -> лети в UserService, /orders/** -> лети в OrderService.2. Безопасность: Проверяет JWT токен один раз на входе.
3. Rate Limiting: "Не больше 10 запросов в секунду от этого юзера".
Пример конфига (
application.yaml):
spring:
cloud:
gateway:
routes:
- id: user-service
uri: lb://USER-SERVICE # lb = Load Balancer (через Eureka)
predicates:
- Path=/users/**
⚡ Итог: Как это работает вместе?
1. Сервисы просыпаются и регистрируются в Eureka.
2. Фронтенд шлет запрос на Gateway.
3. Gateway спрашивает у Eureka адрес нужного сервиса и пересылает запрос.
4. Если сервисам нужно пообщаться между собой, они используют Feign.
#Java #Microservices #SpringCloud #Eureka #Feign
📲 Мы в MAX
👉@BookJava
Please open Telegram to view this post
VIEW IN TELEGRAM
🤝7👍6
📨 Apache Kafka: Нервная система микросервисов
Представьте, что вы заказали пиццу.
🔴 REST подход: Вы стоите у прилавка и смотрите на повара, пока он не закончит. Вы не можете отойти. Если повар уснул - вы застряли.
🔴 Kafka подход: Вы бросаете чек в коробку "Заказы" и уходите по своим делам. Повар возьмет заказ, когда освободится. Когда пицца будет готова, он положит её на стол выдачи, и вы получите уведомление.
Kafka - это распределенный лог событий. Это не просто очередь, это история всего, что произошло в системе.
🧩 Основные понятия
1. Topic (Топик) - Это "папка" или канал, куда падают сообщения. Например:
2. Producer (Продюсер) - Тот, кто пишет сообщения в топик (например, сервис Заказов).
3. Consumer (Консьюмер) - Тот, кто читает сообщения (например, сервис Уведомлений или Склад).
4. Broker (Брокер) - Сервер Kafka, который хранит эти данные.
🚀 Главная фишка: Fire and Forget
Когда
Он просто кидает событие
Сервисы-подписчики (Consumers) разгребут эти сообщения в своем темпе. Если сервис SMS упал, он поднимется через час, прочитает топик с того места, где остановился, и дошлет все смски. Данные не пропадут.
💻 Spring Kafka: Как писать код?
В Spring Boot работа с Kafka максимально упрощена.
1. Настройка (
2. Продюсер (Отправляем сообщение)
Нам нужен бин
3. Консьюмер (Слушаем эфир)
Просто вешаем аннотацию
⚡ Kafka vs RabbitMQ (Коротко)
Частый холивар.
🔴 RabbitMQ - это классическая Очередь. Сообщение прочитали -> оно удалилось. Это "умный брокер, глупые потребители".
🔴 Kafka - это Лог. Сообщения хранятся на диске днями (или вечно). Их можно перечитывать заново (Replay). Это идеально для аналитики и восстановления данных.
🔥 Итог
Kafka позволяет микросервисам быть слабосвязанными (decoupled).
🔴 Сервис А не знает о существовании Сервиса Б.
🔴 Система выдерживает пиковые нагрузки (Kafka просто буферизирует сообщения, пока консьюмеры не разгребут их).
#Java #Kafka #Microservices #SpringCloud #Architecture
📲 Мы в MAX
👉@BookJava
Представьте, что вы заказали пиццу.
Kafka - это распределенный лог событий. Это не просто очередь, это история всего, что произошло в системе.
🧩 Основные понятия
1. Topic (Топик) - Это "папка" или канал, куда падают сообщения. Например:
orders-topic, email-topic.2. Producer (Продюсер) - Тот, кто пишет сообщения в топик (например, сервис Заказов).
3. Consumer (Консьюмер) - Тот, кто читает сообщения (например, сервис Уведомлений или Склад).
4. Broker (Брокер) - Сервер Kafka, который хранит эти данные.
🚀 Главная фишка: Fire and Forget
Когда
OrderService создает заказ, ему плевать, работает ли сейчас сервис отправки SMS или сервис начисления бонусов.Он просто кидает событие
OrderCreated в Kafka и мгновенно возвращает ответ пользователю "Заказ принят".Сервисы-подписчики (Consumers) разгребут эти сообщения в своем темпе. Если сервис SMS упал, он поднимется через час, прочитает топик с того места, где остановился, и дошлет все смски. Данные не пропадут.
💻 Spring Kafka: Как писать код?
В Spring Boot работа с Kafka максимально упрощена.
1. Настройка (
application.yaml)
spring:
kafka:
bootstrap-servers: localhost:9092 # Адрес брокера
consumer:
group-id: my-group # Важно для масштабирования
2. Продюсер (Отправляем сообщение)
Нам нужен бин
KafkaTemplate.
@Service
@RequiredArgsConstructor
public class OrderProducer {
private final KafkaTemplate<String, String> kafkaTemplate;
public void sendOrderEvent(String orderId) {
// Отправляем в топик "orders"
kafkaTemplate.send("orders", orderId);
System.out.println("Сообщение отправлено: " + orderId);
}
}
3. Консьюмер (Слушаем эфир)
Просто вешаем аннотацию
@KafkaListener.
@Service
public class NotificationConsumer {
@KafkaListener(topics = "orders", groupId = "notification_group")
public void listen(String orderId) {
System.out.println("Получено событие для заказа: " + orderId);
// Тут логика отправки email/sms
sendEmail(orderId);
}
}
⚡ Kafka vs RabbitMQ (Коротко)
Частый холивар.
🔥 Итог
Kafka позволяет микросервисам быть слабосвязанными (decoupled).
#Java #Kafka #Microservices #SpringCloud #Architecture
📲 Мы в MAX
👉@BookJava
Please open Telegram to view this post
VIEW IN TELEGRAM
👍8❤2🔥2
🧠 Domain Driven Design: Программируем на языке бизнеса
Главная проблема IT: Разработчики говорят на языке таблиц (
DDD (Предметно-ориентированное проектирование) - это попытка убрать переводчика.
🗣 1. Ubiquitous Language (Единый язык)
Это фундамент. Код должен звучать так же, как речь эксперта.
🔴 Плохо (CRUD-мышление):
• Бизнес: "Клиент сменил адрес доставки".
• Код:
🔴 Хорошо (DDD):
• Код:
Методы должны называться глаголами бизнеса, а не сеттерами.
📦 2. Bounded Context (Ограниченный контекст)
Самая большая ошибка новичка - создать один класс
🔴 Для Продаж: Товар - это цена, описание, картинки.
🔴 Для Склада: Товар - это вес, габариты, номер полки.
🔴 Для Бухгалтерии: Товар - это актив, амортизация, инвентарный номер.
В DDD мы не делаем монстра. Мы создаем разные модели для разных контекстов.
🔴
🔴
Эти модели могут даже иметь разные ID и общаться друг с другом только через события (Kafka).
🏗 3. Tactical DDD (Строительные блоки)
Как писать код внутри контекста?
A. Entity (Сущность)
Объект, у которого есть Identity (ID).
Если у двух людей одинаковое имя, это все равно два разных человека (разные ID).
🔴 Пример:
🔴 Они живут долго, меняют свое состояние, но остаются собой.
B. Value Object (Объект-значение)
Объект, который определяется только своими данными. У него нет ID. Он неизменяем (Immutable).
🔴 Пример:
🔴 Если у меня есть 100 рублей и у вас 100 рублей - это одни и те же 100 рублей. Нам не важно, какая именно это купюра.
🔴 Правило: Не используйте примитивы! Вместо
C. Aggregate (Агрегат)
Это кластер объектов, которые живут и умирают вместе.
🔴 Пример:
🔴 Aggregate Root (Корень): Это главный объект (
🔴 Правило: Извне можно обращаться только к Корню. Нельзя получить ссылку на
🩸 Anemic vs Rich Model (Анемичная vs Богатая модель)
❌ Анемичная (Стандартный Spring):
Класс — это просто мешок с геттерами и сеттерами. Вся логика лежит в
✅ Богатая (DDD):
Логика и валидация живут внутри Сущности. Сервис просто координирует работу.
🔥 Итог
DDD - это сложно, но необходимо для больших проектов.
1. Говорите на языке бизнеса (Ubiquitous Language).
2. Разделяйте модели (Bounded Contexts).
3. Используйте Value Objects вместо примитивов.
4. Прячьте логику внутрь Rich Domain Model.
#Architecture #DDD #DomainDrivenDesign #Java #Microservices
📲 Мы в MAX
👉@BookJava
Главная проблема IT: Разработчики говорят на языке таблиц (
INSERT, Foreign Key, DTO), а бизнес на языке денег и процессов («Провести проводку», «Списать остаток»).DDD (Предметно-ориентированное проектирование) - это попытка убрать переводчика.
🗣 1. Ubiquitous Language (Единый язык)
Это фундамент. Код должен звучать так же, как речь эксперта.
• Бизнес: "Клиент сменил адрес доставки".
• Код:
user.setAddress("New York"); userRepo.save(user);• Код:
user.relocateTo(new Address("New York"));Методы должны называться глаголами бизнеса, а не сеттерами.
📦 2. Bounded Context (Ограниченный контекст)
Самая большая ошибка новичка - создать один класс
Product на всё приложение.В DDD мы не делаем монстра. Мы создаем разные модели для разных контекстов.
SalesContext.ProductWarehouseContext.StockItemЭти модели могут даже иметь разные ID и общаться друг с другом только через события (Kafka).
🏗 3. Tactical DDD (Строительные блоки)
Как писать код внутри контекста?
A. Entity (Сущность)
Объект, у которого есть Identity (ID).
Если у двух людей одинаковое имя, это все равно два разных человека (разные ID).
User, Order.B. Value Object (Объект-значение)
Объект, который определяется только своими данными. У него нет ID. Он неизменяем (Immutable).
Money, Address, Color.String email используйте класс EmailAddress. Там можно спрятать валидацию.C. Aggregate (Агрегат)
Это кластер объектов, которые живут и умирают вместе.
Заказ (Order) + Позиции заказа (OrderItems).Order).OrderItem и изменить его цену напрямую. Вы должны сказать: order.changeItemPrice(...). Это гарантирует целостность данных.🩸 Anemic vs Rich Model (Анемичная vs Богатая модель)
❌ Анемичная (Стандартный Spring):
Класс — это просто мешок с геттерами и сеттерами. Вся логика лежит в
Service.
// Service
public void completeOrder(Long id) {
Order order = repo.findById(id);
if (order.getStatus() != PAID) throw ...
order.setStatus(COMPLETED); // Кто угодно может поменять статус!
repo.save(order);
}
✅ Богатая (DDD):
Логика и валидация живут внутри Сущности. Сервис просто координирует работу.
// Entity
public void complete() {
if (this.status != PAID) throw new DomainException("Не оплачено!");
this.status = COMPLETED;
// Можно даже вернуть событие OrderCompletedEvent
}
// Service
public void completeOrder(Long id) {
Order order = repo.findById(id);
order.complete(); // Вся бизнес-логика внутри
repo.save(order);
}
🔥 Итог
DDD - это сложно, но необходимо для больших проектов.
1. Говорите на языке бизнеса (Ubiquitous Language).
2. Разделяйте модели (Bounded Contexts).
3. Используйте Value Objects вместо примитивов.
4. Прячьте логику внутрь Rich Domain Model.
#Architecture #DDD #DomainDrivenDesign #Java #Microservices
📲 Мы в MAX
👉@BookJava
Please open Telegram to view this post
VIEW IN TELEGRAM
👍7❤1👎1