Развенчиваем распространенный миф про Java Garbage Collector
Миф: мне НЕ нужно заниматься управлением памятью в Java, потому что GC все делает за меня
Garbage Collector (GC) действительно очищает неиспользуемые объекты, на которые больше нет активных ссылок. Благодаря этому в Java не нужно вручную освобождать память, как в C++.
Несколько предпосылок, из которых обычно рождается этот тезис:
- В языках с ручным управлением памятью (C, C++) утечки очевидны: если забыл вызвать
- В Java GC работает автоматически, поэтому кажется, что он сам решает все проблемы.
- «Ну раз есть GC, значит о памяти можно больше не думать» - типичная ошибка.
GC удаляет только те объекты, на которые больше нет активных ссылок. Если объект остается доступным, но фактически уже не используется, он будет занимать память до завершения приложения.
* Несколько случаев утечек памяти
[1] Статические коллекции (заполняем, но не очищаем)
Если создать
Через пару минут -
[2] Переменные потока (
Объекты, сохраненные в
Поток завершится, но память останется занятой, потому что
[3] Внутренние классы и «утекшие» ссылки
Если анонимный класс или lambda-ссылка захватывает внешний объект, это может мешать GC освободить его.
Миф развенчан. GC не всесилен, и даже с ним придется учиться правильно работать с памятью в Java.
👉 Java Portal
Миф: мне НЕ нужно заниматься управлением памятью в Java, потому что GC все делает за меня
Garbage Collector (GC) действительно очищает неиспользуемые объекты, на которые больше нет активных ссылок. Благодаря этому в Java не нужно вручную освобождать память, как в C++.
Несколько предпосылок, из которых обычно рождается этот тезис:
- В языках с ручным управлением памятью (C, C++) утечки очевидны: если забыл вызвать
free(), память потеряна навсегда.- В Java GC работает автоматически, поэтому кажется, что он сам решает все проблемы.
- «Ну раз есть GC, значит о памяти можно больше не думать» - типичная ошибка.
GC удаляет только те объекты, на которые больше нет активных ссылок. Если объект остается доступным, но фактически уже не используется, он будет занимать память до завершения приложения.
* Несколько случаев утечек памяти
[1] Статические коллекции (заполняем, но не очищаем)
Если создать
static List и постоянно добавлять в него объекты, GC их никогда не освободит, потому что статические поля живут в течение всего времени работы приложения.public class MemoryLeak {
private static final List<byte[]> cache = new ArrayList<>();
public static void main(String[] args) {
while (true) {
cache.add(new byte[10 * 1024 * 1024]);
System.out.println("Added 10MB to the cache. Used memory: " +
(Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory()) / (1024 * 1024) + "MB");
}
}
}Через пару минут -
OutOfMemoryError.[2] Переменные потока (
ThreadLocal)Объекты, сохраненные в
ThreadLocal, привязаны к потоку, а в пуле потоков могут жить дольше, чем нужно.public class ThreadLocalLeak {
private static final ThreadLocal<byte[]> threadLocalData = new ThreadLocal<>();
public static void main(String[] args) {
ExecutorService executor = Executors.newFixedThreadPool(5);
for (int i = 0; i < 10; i++) {
executor.execute(() -> {
threadLocalData.set(new byte[10 * 1024 * 1024]); // 10MB per thread
System.out.println("Memory occupied by the thread!");
});
}
executor.shutdown();
}
}Поток завершится, но память останется занятой, потому что
ThreadLocal не очищается автоматически.[3] Внутренние классы и «утекшие» ссылки
Если анонимный класс или lambda-ссылка захватывает внешний объект, это может мешать GC освободить его.
public class InnerClassLeak {
private String data = "Very important data";
public void createAnonymousClass() {
Runnable task = new Runnable() {
@Override
public void run() {
System.out.println("Using: " + data);
}
};
new Thread(task).start();
}
}task держит ссылку на data, и даже если InnerClassLeak больше не используется, GC не сможет очистить объект.Миф развенчан. GC не всесилен, и даже с ним придется учиться правильно работать с памятью в Java.
Please open Telegram to view this post
VIEW IN TELEGRAM
👍4🔥2
Spring Boot: можно использовать
✅ Если что-то не совпадает, приложение не запустится
✅ Это можно комбинировать с инструментами миграций, например Flyway или Liquibase
👉 Java Portal
spring.jpa.hibernate.ddl-auto=validate, чтобы проверять случайные изменения схемы в продакшене.Please open Telegram to view this post
VIEW IN TELEGRAM
👍4❤2
Твоя база данных не медленная.
Медленный у тебя цикл обратной связи.
Один сеньор написал практический гайд, где показано, как разработчику на Java анализировать производительность PostgreSQL из Quarkus-приложения с помощью
👉 Java Portal
Медленный у тебя цикл обратной связи.
Один сеньор написал практический гайд, где показано, как разработчику на Java анализировать производительность PostgreSQL из Quarkus-приложения с помощью
pg_stat_statements, hypopg и PostgreSQL MCP-сервера через IBM Bob.Please open Telegram to view this post
VIEW IN TELEGRAM
The-Main-Thread
How to Find Database Performance Problems with Quarkus, PostgreSQL MCP, and AI
A practical guide for Java developers to analyze slow queries, missing indexes, and database health using Quarkus and IBM Bob
👍6❤1
Совет: держите код рядом. Следите, чтобы вычисление было как можно ближе к месту, где используется это значение. Это улучшает читаемость и удобство рефакторинга, снижает вероятность багов и делает код более устойчивым к ошибкам.
👉 Java Portal
Please open Telegram to view this post
VIEW IN TELEGRAM
❤6
Совет по Java: используйте
👉 Java Portal
Collectors.groupingBy(...), чтобы группировать результаты стрима по классификатору.Please open Telegram to view this post
VIEW IN TELEGRAM
👀4
Пессимистическая vs оптимистическая блокировка
Частый вопрос на собеседованиях по backend.
[Пессимистическая блокировка]: -
- Запись недоступна другим потокам, пока текущий поток не закончит с ней работу.
- Даже чтение данных другими потоками невозможно, пока блокировка не снята.
- Пример:
- Минусы: снижает производительность при высоком уровне конкуренции.
- Когда использовать: когда критически важна точность данных.
[Оптимистическая блокировка]: -
- Не блокирует данные при чтении, но при сохранении проверяет версию записи.
- Пример: аннотация
- Если версия изменилась другим потоком, выбрасывается
- Минусы: нужно разруливать конфликты.
- Когда использовать: когда чтений много, а вероятность конфликтов низкая.
Какой подход ты чаще используешь в своих проектах?
👉 Java Portal
Частый вопрос на собеседованиях по backend.
[Пессимистическая блокировка]: -
- Запись недоступна другим потокам, пока текущий поток не закончит с ней работу.
- Даже чтение данных другими потоками невозможно, пока блокировка не снята.
- Пример:
EntityManager.lock(entity, LockModeType.PESSIMISTIC_WRITE) в JPA.- Минусы: снижает производительность при высоком уровне конкуренции.
- Когда использовать: когда критически важна точность данных.
[Оптимистическая блокировка]: -
- Не блокирует данные при чтении, но при сохранении проверяет версию записи.
- Пример: аннотация
@Version в JPA (колонка для хранения версии).- Если версия изменилась другим потоком, выбрасывается
OptimisticLockException.- Минусы: нужно разруливать конфликты.
- Когда использовать: когда чтений много, а вероятность конфликтов низкая.
Какой подход ты чаще используешь в своих проектах?
Please open Telegram to view this post
VIEW IN TELEGRAM
👍3
На Stepik добавили курс «Linux с нуля»
Этот курс закрывает всю обязательную Linux-базу для работы в IT. Подойдёт для:
Внутри 20+ модулей: от установки Linux и работы с файлами до сетей, прав, дисков, процессов, автоматизации на Bash и многого другого. Всё сразу закрепляется на практике (200+ заданий с автопроверкой)
Материал подаётся понятным языком, шаг за шагом, на реальных примерах и с наглядными схемами
После прохождения вы получите сертификат, который можно добавить в резюме.
В ближайшие 48ч курс доступен со скидкой 30% по промокоду «
Этот курс закрывает всю обязательную Linux-базу для работы в IT. Подойдёт для:
- разработчиков
- девопсов и админов
- специалистов по данным и ML
- специалистов поддержки и сопровождения
- тестировщиков и безопасников
Внутри 20+ модулей: от установки Linux и работы с файлами до сетей, прав, дисков, процессов, автоматизации на Bash и многого другого. Всё сразу закрепляется на практике (200+ заданий с автопроверкой)
Материал подаётся понятным языком, шаг за шагом, на реальных примерах и с наглядными схемами
После прохождения вы получите сертификат, который можно добавить в резюме.
В ближайшие 48ч курс доступен со скидкой 30% по промокоду «
PORTAL30»: открыть курс на Stepik❤2
Задумывался, как работают ActiveMQ и RabbitMQ?
В основе у них producer/consumer паттерн.
Как его реализовать?
Это базовая схема обмена данными между несколькими потоками. Поток-продюсер отправляет объекты на условную обработку, а потоки-консьюмеры асинхронно их получают и обрабатывают.
Общий вид решения такой:
продюсер кладет объекты в специальную коллекцию, буфер. Когда консьюмер освобождается, он запрашивает из буфера один объект. Если буфер пустой, консьюмер блокируется и ждёт; если буфер переполнен, ждёт продюсер.
Реализовать этот паттерн можно по-разному.
Самый правильный способ для продакшена: брать готовую реализацию из стандартной библиотеки, например
На собеседованиях обычно просят написать реализацию с нуля. На картинке показан один из вариантов.
Модификатор
👉 Java Portal
В основе у них producer/consumer паттерн.
Как его реализовать?
Это базовая схема обмена данными между несколькими потоками. Поток-продюсер отправляет объекты на условную обработку, а потоки-консьюмеры асинхронно их получают и обрабатывают.
Общий вид решения такой:
продюсер кладет объекты в специальную коллекцию, буфер. Когда консьюмер освобождается, он запрашивает из буфера один объект. Если буфер пустой, консьюмер блокируется и ждёт; если буфер переполнен, ждёт продюсер.
Реализовать этот паттерн можно по-разному.
Самый правильный способ для продакшена: брать готовую реализацию из стандартной библиотеки, например
BlockingQueue.На собеседованиях обычно просят написать реализацию с нуля. На картинке показан один из вариантов.
Модификатор
synchronized гарантирует, что в один момент времени выполняется только один из методов и только одним потоком. Этого достаточно для корректной работы, пока буфер не пуст и не переполнен. Когда буфер пустой или полный, управление явно передаётся продюсеру или консьюмеру соответственно через notify() и wait().Please open Telegram to view this post
VIEW IN TELEGRAM
❤3
Почему твой бэкенд начинает тупить на 10k юзеров, хотя на 100 всё летало
Обычно причина одна из этих:
1️⃣ N+1 запросы
→ один запрос незаметно триггерит сотню походов в БД
2️⃣ Нет индексов
→ на каждом поиске полный проход по таблице
3️⃣ Тяжелая работа синхронно
→ письма, обработка картинок прямо внутри request cycle
4️⃣ Нет стратегии кеширования
→ один и тот же запрос в БД выполняется 1000 раз
5️⃣ Единая точка отказа
→ один инстанс БД тащит на себе вообще всё
Приложение не стало внезапно медленным.
Оно просто доросло до момента, когда плохие решения стали заметны.
👉 Java Portal
Обычно причина одна из этих:
→ один запрос незаметно триггерит сотню походов в БД
→ на каждом поиске полный проход по таблице
→ письма, обработка картинок прямо внутри request cycle
→ один и тот же запрос в БД выполняется 1000 раз
→ один инстанс БД тащит на себе вообще всё
Приложение не стало внезапно медленным.
Оно просто доросло до момента, когда плохие решения стали заметны.
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥8❤2
Совет по Java:
Избавляет от
👉 Java Portal
Stream.toArray(Type[]::new) это аккуратный и типобезопасный способ получить массив из стрима.Избавляет от
Object[] и ручных кастов.Please open Telegram to view this post
VIEW IN TELEGRAM
👍4❤2
Вопрос для интервью по Java Spring Boot:
С тех пор как AI начал активно залезать в мир разработки, паттерны на собеседованиях меняются.
Тренд, который я недавно наблюдал: дают сниппет кода, и ты должен его проревьюить и отрефакторить логику под прод.
[Дано]:
Команда написала логику отправки письма после регистрации пользователя. На проде иногда бывает ситуация: письма приходят, а пользователя в базе нет. Найди проблему и почини:
[Задачи]:
- Объясни, в каком сценарии письмо будет отправлено, но пользователь не сохранится
- Исправь код так, чтобы событие обрабатывалось только после того, как пользователь сохранён
Код:
👉 Java Portal
@Transactional + @EventListenerС тех пор как AI начал активно залезать в мир разработки, паттерны на собеседованиях меняются.
Тренд, который я недавно наблюдал: дают сниппет кода, и ты должен его проревьюить и отрефакторить логику под прод.
[Дано]:
Команда написала логику отправки письма после регистрации пользователя. На проде иногда бывает ситуация: письма приходят, а пользователя в базе нет. Найди проблему и почини:
[Задачи]:
- Объясни, в каком сценарии письмо будет отправлено, но пользователь не сохранится
- Исправь код так, чтобы событие обрабатывалось только после того, как пользователь сохранён
Код:
@Service
@RequiredArgsConstructor
public class UserService {
private final UserRepository userRepository;
private final ApplicationEventPublisher eventPublisher;
@Transactional
public void register(UserDto dto) {
User user = new User(https://dto.email());
https://userRepository.save(user);
eventPublisher.publishEvent(new UserRegisteredEvent(user));
}
}
@Component
@RequiredArgsConstructor
public class EmailListener {
private final EmailSender emailSender;
private final SomeOtherService someOtherService;
@EventListener
public void onUserRegistered(UserRegisteredEvent event) {
emailSender.sendWelcome(event.user().getEmail());
someOtherService.doSomething();
}
}
Please open Telegram to view this post
VIEW IN TELEGRAM