Java Portal | Программирование
12.8K subscribers
1.23K photos
104 videos
38 files
1.19K links
Присоединяйтесь к нашему каналу и погрузитесь в мир для Java-разработчика

Связь: @devmangx

РКН: https://clck.ru/3H4WUg
Download Telegram
Совет: в REST-сервисах ты часто получаешь только ID сущностей. И не всегда нужно тащить их из БД.

Чтобы проставить связь (по сути, foreign key в базе), можно “создать” ссылку на сущность через getReferenceById(id) без SELECT и использовать её, чтобы задать relation.

@Service
public SomeService {

@Autowired
private ProductRepository productRepository;

@Autowired
private CategoryRepository featureRepository;

public void setCategory(long productId, long categoryId) {
Product product = productRepository.findById(productId).orElseThrow(); // executes SELECT
Category category = categoryRepository.getReferenceById(categoryId); // does not execute a SELECT
product.setCategory(category);
productRepository.save(product);
}

}


👉 Java Portal
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥7👍5
Spring Boot: можно реализовать оптимистическую блокировку через аннотацию @Version.

Идеально подходит, когда:

- чтения происходят часто
- записи сравнительно редкие
- конфликты случаются нечасто

Не лучший вариант, когда:

- конфликты происходят часто
- нужна строгая сериализация

public class Account {

@Id
private Long id;

private BigDecimal balance;

@Version
private Long version;
}


Когда читаем, Entity загружается с текущей версией. Допустим, version = 2.

Когда делаем update, учитывается запись с version = 2.

Если за это время другая транзакция уже успела её обновить, version уже не 2, апдейт не затронет ни одной строки, и Hibernate кинет OptimisticLockException.

То есть конфликт ловится без блокировки строки.

👉 Java Portal
Please open Telegram to view this post
VIEW IN TELEGRAM
👍2
Реальный event-driven проект на Core Java 21, Apache Kafka (KRaft mode) и PostgreSQL (в Docker).

Проект показывает, как современные распределенные системы обрабатывают заказы асинхронно через Kafka, без блокировок и ожидания между сервисами.

https://github.com/FlexiDriod/kafka-order-processing-system

👉 Java Portal
Please open Telegram to view this post
VIEW IN TELEGRAM
🤣2😁1
Spring Boot: отдавай предпочтение статическим utility-классам вместо Spring-бинов только если они не хранят состояние, у них нет зависимостей и их не нужно мокать или оборачивать в AOP.

public class CacheUtils {
private static Map<String, String> cache = new HashMap<>();

...
}

// Кэш как выше разделяется между всеми потоками и может случайно изменяться

@Component
@Scope("request")
public class CacheService {
private Map<String, String> cache = new HashMap<>();

...
}

// Если оформить это как бин, можно задать ему конкретный scope.

// Spring лучше управляет объектами с состоянием.


👉 Java Portal
Please open Telegram to view this post
VIEW IN TELEGRAM
👍3👀3💊31
Большинство разработчиков в Spring знают, как доставать query-параметры и path variables.

Но часто забывают, что HTTP-хедеры можно вытаскивать так же просто. В Spring для этого есть аккуратный @RequestHeader.

Можно в одну строку получить User-Agent, Authorization или любой кастомный хедер. И не нужно лезть в HttpServletRequest.

Пример:

▪️Добавляешь в метод контроллера @RequestHeader("User-Agent") String userAgent
▪️Spring сам подхватывает значение из заголовка
▪️Ты получаешь его сразу параметром метода

Нужно сделать опциональным?
Просто добавь required = false.

Нужен дефолт?

▪️Используй defaultValue = "unknown".
▪️Очень полезно для логирования, аналитики, A/B тестов или фичефлагов, завязанных на заголовки.
▪️Выручает при работе с API-клиентами, которые шлют кастомные заголовки.
▪️Чище, короче и легче тестируется.
▪️Чем меньше зависимостей от servlet API, тем лучше.

👉 Java Portal
Please open Telegram to view this post
VIEW IN TELEGRAM
👍19
...Script


👉 Java Portal
Please open Telegram to view this post
VIEW IN TELEGRAM
😁104
Java-совет: чтобы не ловить оверхед на boxing/unboxing, используй стримы примитивов (IntStream/LongStream/DoubleStream) вместо Stream<Integer>/Stream<Long>/Stream<Double>.

Создаём объекты Integer для каждого значения:

Stream<Integer> boxed = Stream.of(1, 2, 3, 4, 5);
int sumBoxed = boxed.reduce(0, Integer::sum);


Работаем напрямую с int:

int sumPrimitive = IntStream.of(1, 2, 3, 4, 5).sum();


Другие примитивные стримы:

LongStream, DoubleStream

👉 Java Portal
Please open Telegram to view this post
VIEW IN TELEGRAM
3👍3