ru
Feedback
Java | Фишки и трюки

Java | Фишки и трюки

Открыть в Telegram

Java: примеры кода, интересные фишки и полезные трюки Купить рекламу: https://telega.in/c/java_tips_and_tricks ✍️По всем вопросам: @Pascal4eg Менеджер по рекламе: @shmyzna

Больше
6 960
Подписчики
+1024 часа
+457 дней
+7930 день
Архив постов
⌨️ Java Concurrency API: Основные классы Современные приложения все чаще требуют выполнения нескольких задач одновременно, и для этого Java предоставляет мощный инструмент — Concurrency API (набор классов и интерфейсов). Этот API позволяет разработчикам легко реализовывать многопоточность, управлять потоками и синхронизировать действия между ними, что значительно увеличивает производительность и отзывчивость приложений. Основные классы: 1. Thread - Класс для создания и управления потоками. Вы можете создать новый поток, реализовав интерфейс Runnable или расширив класс Thread. 2. Executor - Интерфейс для управления потоками и выполнения задач. Позволяет абстрагироваться от управления потоками, сосредоточившись на логике приложения. - ExecutorService: расширение Executor, управляющее жизненным циклом потоков. 3. Future - Позволяет получать результаты из асинхронных задач. Используется в связке с ExecutorService для выполнения задач в фоновом режиме. 4. CountDownLatch - Синхронизирует потоки, позволяя одному или нескольким потокам ждать завершения других потоков перед продолжением работы. 5. CyclicBarrier - Используется для синхронизации группы потоков. Позволяет потоку ждать, пока все другие не достигнут определенной точки. 6. Semaphore - Контролирует доступ к ресурсу, предоставляя определенное количество разрешений для потоков. 7. BlockingQueue - Интерфейс, предоставляющий безопасный способ обмена данными между потоками при помощи очередей. Реализации включают ArrayBlockingQueue, LinkedBlockingQueue и другие. #java #ConcurrencyAPI #Thread

Собрали подборку IT-каналов для тех, кто интересуется тестированием, разработкой, автотестами и карьерой в IT. Внутри: QA и A
Собрали подборку IT-каналов для тех, кто интересуется тестированием, разработкой, автотестами и карьерой в IT. Внутри: QA и AQA - практика, собеседования, рабочие заметки и разборы Java и Python - задачи, тесты, полезные фишки и материалы для разработчиков SQL и аналитика данных - запросы, работа с данными и прикладные разборы Программирование и IT-карьера - каналы для тех, кто развивается в профессии и хочет лучше понимать рынок Можно подписаться на всю подборку сразу или выбрать только те каналы, которые подходят под ваши цели. Забрать папку

⌨️ Desktop Класс Desktop используется для взаимодействия с приложениями операционной системы, такими как веб-браузер, почтовый клиент, просмотрщик изображений и т.д. Этот класс входит в пакет java.awt и позволяет, например, открыть веб-ссылку в браузере или отправить письмо через почтовый клиент. Основные возможности класса Desktop: ✔️ browse(URI uri) – открывает URI (например, веб-страницу) в браузере. ✔️ open(File file) – открывает файл в приложении, ассоциированном с его типом (например, текстовый файл в текстовом редакторе). ✔️ edit(File file) – открывает файл в режиме редактирования (если доступно). ✔️ mail(URI mailtoURI) – открывает почтовое приложение с указанным URI. ✔️ print(File file) – отправляет файл на печать. Пример:

import java.awt.*;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;

public class DesktopExample {
    public static void main(String[] args) throws IOException, URISyntaxException {
        // Проверяем, поддерживает ли система класс Desktop
        if (Desktop.isDesktopSupported()) {
            Desktop desktop = Desktop.getDesktop();
            // Пример: Открытие веб-страницы
            URI uri = new URI("http://www.google.com");
            desktop.browse(uri);
        } else {
            System.out.println("Класс Desktop не поддерживается на этой системе.");
        }
    }
}
Проверка Desktop.isDesktopSupported() обязательна, так как класс может не поддерживаться на некоторых системах. #java #Desktop

⌨️ Почему Stream.parallel() может сделать код медленнее Когда разработчики впервые видят:

list.parallelStream()
возникает мысль: О, бесплатная многопоточность 🚀 Но на практике parallelStream() очень часто: ❗ НЕ ускоряет код ❗ делает его медленнее ❗ а иногда ещё и ломает логику 📦 Как это работает

list.parallelStream()
    .map(this::process)
    .toList();
Java разбивает задачи между потоками через ForkJoinPool.commonPool(). 🔥 Когда это реально помогает ✔️ CPU-heavy вычисления ✔️ большие объёмы данных ✔️ независимые операции Например: ✅ обработка миллионов чисел ✅ сложные математические вычисления ✅ image processing ⚠️ Когда будет хуже ❌ Маленькие коллекции

List.of(1, 2, 3)
    .parallelStream()
👉 накладные расходы на потоки будут больше пользы ❌ I/O операции

users.parallelStream()
     .map(this::callApi)
👉 потоки начинают ждать сеть/БД 👉 ForkJoinPool забивается ❌ Общие mutable-данные

List<String> result = new ArrayList<>();

list.parallelStream()
    .forEach(result::add);
💣 Race condition detected 🙂 ❌ Неочевидный порядок

parallelStream().forEach(System.out::println);
Порядок вывода будет случайным. 🧠 Самая скрытая проблема parallelStream() использует ОБЩИЙ pool потоков. То есть: ▸ твой код ▸ библиотека ▸ другой модуль приложения могут конкурировать за одни и те же потоки. 👉 Особенно весело в backend-приложениях. 🚀 Иногда обычный stream быстрее Почему? Потому что parallel: ▸ делит задачи ▸ синхронизирует результаты ▸ координирует потоки Это тоже стоит времени. ✅ Хорошее правило Используй parallelStream() только если: ▸ операция тяжёлая ▸ данных много ▸ нет shared mutable state ▸ ты реально измерил performance

⌨️ EnumMap — это реализация Map, которая использует в качестве ключей исключительно Enum. Внутренне EnumMap использует массив для хранения значений, где индекс массива соответствует порядковому номеру (ordinal()) элемента перечисления. Это делает его быстрее, чем хэш-таблицы (HashMap) и более экономным по памяти. EnumMap хранит ключи в порядке их объявления в перечислении, что отличает его от большинства других Map (кроме LinkedHashMap). Так как ключи строго ограничены перечислением, это снижает вероятность ошибок при разработке. 🔍 Пример:

import java.util.EnumMap;

enum Action {
    START, STOP, PAUSE
}

public class EnumMapExample {
    public static void main(String[] args) {
        EnumMap<Action, Runnable> actionMap = new EnumMap<>(Action.class);

        // Определяем поведение для каждого значения Enum
        actionMap.put(Action.START, () -> System.out.println("Starting the process..."));
        actionMap.put(Action.STOP, () -> System.out.println("Stopping the process..."));
        actionMap.put(Action.PAUSE, () -> System.out.println("Pausing the process..."));

        // Пример вызова
        Action currentAction = Action.START;
        actionMap.get(currentAction).run();
    }
}
#Java #EnumMap

⌨️ ThreadLocal - удобство, которое легко превращается в утечку памяти ThreadLocal выглядит как магия: У каждого потока будет своё собственное значение И это правда. Но именно из-за этого он часто становится источником очень неприятных багов в production. 📦 Как работает ThreadLocal

ThreadLocal<String> currentUser = new ThreadLocal<>();

currentUser.set("Alex");
Теперь в каждом потоке хранится своё значение:

System.out.println(currentUser.get());
🔥 Где это используют * Spring Security * Hibernate * логирование (`traceId`) * транзакции * хранение request context 🤔 Кажется идеально. В чём проблема? Проблема начинается с thread pool. ⚠️ Главная ловушка

ThreadLocal<String> local = new ThreadLocal<>();

executor.submit(() -> {
    local.set("DATA");
});
Поток из пула НЕ уничтожается после задачи. 👉 Значение остаётся висеть внутри потока. 💣 Что это вызывает * утечки памяти * неожиданные данные между запросами * "рандомные" баги в серверных приложениях 😨 Самое неприятное Даже если сам ThreadLocal удалён GC, значение внутри потока может продолжать жить. Почему? Потому что внутри Thread: * ключ (`ThreadLocal`) хранится как weak reference * а значение - как strong reference 👉 Ключ умер 👉 Значение осталось Классическая hidden memory leak. ✅ Правильный подход ВСЕГДА чистить ThreadLocal:

try {
    local.set("DATA");

    // работа
} finally {
    local.remove();
}
🚀 Почему это критично в Spring / Tomcat Сервер использует thread pool: * один поток обслуживает тысячи запросов * забытый ThreadLocal живёт очень долго Отсюда: * рост памяти * странное поведение пользователей * “невоспроизводимые” баги 🧠 Интересный факт ThreadLocal не хранит данные “в себе”. Наоборот: 👉 каждый Thread хранит ThreadLocalMap То есть данные лежат внутри потока, а не объекта ThreadLocal.

⌨️ Сравнение объектов с помощью Comparator и Comparable В Java для сортировки объектов используются два ключевых интерфейса: Comparable и Comparator. Понимание их отличий и применения поможет вам эффективно управлять сортировкой данных в коллекциях. 📚 Краткие определения: - Comparable: - Используется, когда класс должен иметь естественный порядок. Например, мы можем сортировать людей по возрасту. - Сигнатура метода: int compareTo(T o). Пример:

  class Person implements Comparable<Person> {
      private String name;
      private int age;

      public Person(String name, int age) {
          this.name = name;
          this.age = age;
      }

      public int getAge() {
          return age;
      }

      @Override
      public int compareTo(Person other) {
          return Integer.compare(this.age, other.age); // Сравнение по возрасту
      }
  }
- Comparator: - Используется для создания внешних стратегий сравнения, позволяя определять несколько способов сортировки. Подходит для сортировки объектов разных классов. - Сигнатура метода: int compare(T o1, T o2). Пример:

  import java.util.Comparator;

  class NameComparator implements Comparator<Person> {
      @Override
      public int compare(Person p1, Person p2) {
          return p1.name.compareTo(p2.name); // Сравнение по имени
      }
  }
⚖️ Когда использовать: - Comparable: - Используйте, если у класса есть естественный порядок. Например, чтобы сортировать людей по возрасту. - Сортировка с помощью Collections.sort() или Arrays.sort() будет проще. Пример использования:

  List<Person> people = Arrays.asList(new Person("Alice", 30), new Person("Bob", 25));
  Collections.sort(people); // Сортировка по возрасту с использованием Comparable
- Comparator: - Идеален для ситуаций, когда необходимо изменить порядок сортировки или сравнивать объекты разных классов. Пример использования:

  List<Person> people = Arrays.asList(new Person("Alice", 30), new Person("Bob", 25));
  Collections.sort(people, new NameComparator()); // Сортировка по имени с использованием Comparator
#java #Comparator #Comparable

⌨️ Чем абстрактный класс отличается от интерфейса? В каких случаях следует использовать абстрактный класс, а в каких интерфейс? ✔️ В Java класс может одновременно реализовать несколько интерфейсов, но наследоваться только от одного класса. ✔️ Абстрактные классы используются только тогда, когда присутствует тип отношений «is a» (является). Интерфейсы могут реализоваться классами, которые не связаны друг с другом. ✔️ Абстрактный класс - средство, позволяющее избежать написания повторяющегося кода, инструмент для частичной реализации поведения. Интерфейс - это средство выражения семантики класса, контракт, описывающий возможности. Все методы интерфейса неявно объявляются как public abstract или (начиная с Java 8) default - методами с реализацией по-умолчанию, а поля - public static final. ✔️ Интерфейсы позволяют создавать структуры типов без иерархии. ✔️ Наследуясь от абстрактного, класс «растворяет» собственную индивидуальность. Реализуя интерфейс, он расширяет собственную функциональность. Абстрактные классы содержат частичную реализацию, которая дополняется или расширяется в подклассах. При этом все подклассы схожи между собой в части реализации, унаследованной от абстрактного класса, и отличаются лишь в части собственной реализации абстрактных методов родителя. Поэтому абстрактные классы применяются в случае построения иерархии однотипных, очень похожих друг на друга классов. В этом случае наследование от абстрактного класса, реализующего поведение объекта по умолчанию может быть полезно, так как позволяет избежать написания повторяющегося кода. Во всех остальных случаях лучше использовать интерфейсы. #java #interface #abstract

🔴 Завтра тестовое собеседование с Java-разработчиком 13 мая(уже завтра!) в 19:00 по мск приходи онлайн на открытое собеседов
🔴 Завтра тестовое собеседование с Java-разработчиком 13 мая(уже завтра!) в 19:00 по мск приходи онлайн на открытое собеседование, чтобы посмотреть на настоящее интервью на Middle Java-разработчика. Как это будет: 📂 Виктор Анохин, старший разработчик из WildBerries, будет задавать реальные вопросы и задачи разработчику-добровольцу 📂 Виктор будет комментировать каждый ответ респондента, чтобы дать понять чего от вас ожидает собеседующий на интервью 📂 В конце можно будет задать любой вопрос Виктору Это бесплатно. Эфир проходит в рамках менторской программы от ШОРТКАТ для Java-разработчиков, которые хотят повысить свой грейд, ЗП и прокачать скиллы. Переходи в нашего бота, чтобы получить ссылку на эфир → @shortcut_sh_bot Реклама. О рекламодателе.

🛠 Разница между StringBuilder и StringBuffer И StringBuilder, и StringBuffer используются для работы со строками, но между ними есть важные отличия. ➡️ Основные различия: ✔️ StringBuilder быстрее, но не потокобезопасен. ✔️ StringBuffer потокобезопасен, но медленнее из-за синхронизации. ✔️ Пример использования StringBuilder:
StringBuilder sb = new StringBuilder("Hello");
sb.append(" World");
System.out.println(sb); // Hello World
✔️ Пример использования StringBuffer:
StringBuffer sb = new StringBuffer("Hello");
sb.append(" World");
System.out.println(sb); // Hello World
💡 Совет: Если ваш код выполняется в одном потоке, используйте StringBuilder для лучшей производительности. #java #stringbuilder #stringbuffer

⚠️ Внимание: Опубликована слитая недавно База из 1000+ топовых курсов и материалов для айтишников Отсортировали их для вашего удобства и выложили в телеграм-каналы по категориям: 🤖 Нейросети – 855+ материалов 🖥 Python — 1558+ материалов 👩‍💻 Frontend — 1241+ материалов 👩‍💻 Backend — 1095+ материалов ⚙️ Программы — 978+ материалов 📚 Книги по IT — 779+ материалов ⌛️ Ссылки будут активны 48 часов, сохрани себе, чтобы не потерять!

⌨️ Arrays.deepToString() Метод deepToString из класса Arrays используется для создания строкового представления многомерных массивов (например, массивов массивов). Он обходит каждый уровень вложенности массива и выводит его элементы в виде строки. Это удобно для работы с многомерными массивами, так как стандартный метод toString не раскрывает их структуру. Пример использования:

int[][] array = {{1, 2, 3}, {4, 5, 6}};
System.out.println(Arrays.deepToString(array));
// [[1, 2, 3], [4, 5, 6]]
Этот метод работает рекурсивно, обеспечивая полное отображение структуры массива любой вложенности. #java #Arrays #deepToString

📌 Что такое volatile и как оно работает? Ключевое слово volatile в Java используется для переменных, к которым обращаются несколько потоков. Оно гарантирует, что изменения переменной одним потоком будут видны другим. ✔️ Основные особенности: 1️⃣ Обеспечивает чтение переменной из основной памяти, а не из кэша потока. 2️⃣ Гарантирует видимость изменений между потоками. Пример:
class SharedObject {
    volatile int count = 0;
}
💡 Ограничение: volatile не гарантирует атомарность операций. Для этого используйте synchronized или классы из java.util.concurrent. #java #volatile #multithreading

Изучаете программирование, или хотите стать классным айтишником с высокой ЗП? Пока многие паникуют из-за бума ИИ, вы можете с
Изучаете программирование, или хотите стать классным айтишником с высокой ЗП? Пока многие паникуют из-за бума ИИ, вы можете стремительно войти в сферу IT при помощи сборки "IT в деле" которую мы составили. Специально для вас разбили нашу сборку мини папки по конкретным тематикам:) Что внутри сборки: - Каналы о программировании и разработке: Python, CSS, HTML, Java и Swift - GameDev и каналы о QA тестироварии, создание игр, помощь для ваших проектов и реальный опыт от тестировщиков 💍 - Авторские каналы и вайбкодинг: IT & AI, простые заметки с проектов, работа с нейросетями, и как они реально влияют на индустрию 🔤 - Информационная безопасность: многое узнаете о работе хакеров и о том, как защитить себя в этом интернет-хаосе. 🔫 Для каждого здесь что-то найдётся! Выберите интересные вам каналы или подпишитесь на всю сборку, и найдите то, что нужно вам для следующего шага в IT.

⌨️ Почему `try-with-resources` это больше, чем "синтаксический сахар" Многие воспринимают try-with-resources как просто удобную замену finally. Но под капотом есть важная деталь, о которой часто забывают - подавленные исключения (suppressed exceptions). 📦 Классический код

try {
    InputStream is = new FileInputStream("file.txt");
    // работа с потоком
} finally {
    is.close();
}
❗ Проблема: если в try и в close() произойдут исключения - одно из них потеряется. ✅ try-with-resources

try (InputStream is = new FileInputStream("file.txt")) {
    // работа с потоком
}
👉 Java сама закроет ресурс и корректно обработает исключения 🔥 Вот где магия

try (MyResource r = new MyResource()) {
    throw new RuntimeException("Ошибка в try");
}
И допустим, close() тоже кидает исключение:

@Override
public void close() {
    throw new RuntimeException("Ошибка при закрытии");
}
🤯 Что произойдёт? * Основное исключение: "Ошибка в try" * Исключение из close() НЕ потеряется * Оно попадёт в suppressed exceptions 📌 Как их получить

try (MyResource r = new MyResource()) {
    ...
} catch (Exception e) {
    for (Throwable t : e.getSuppressed()) {
        t.printStackTrace();
    }
}
⚠️ Почему это важно Если не знаешь про suppressed exceptions: * можно потерять важную информацию об ошибках * сложнее дебажить ресурсы (файлы, сокеты, DB) 🧠 Кастомные ресурсы Чтобы использовать try-with-resources, класс должен реализовывать AutoCloseable:

class MyResource implements AutoCloseable {
    @Override
    public void close() {
        // cleanup
    }
}
👉 Даже свои классы можно встроить в этот механизм 🚀 Маленький лайфхак (Java 9+) Можно объявить ресурс заранее:

InputStream is = new FileInputStream("file.txt");

try (is) {
    // используем
}
📌 Вывод try-with-resources это не только удобство, а ещё и правильная модель обработки исключений, которую вручную реализовать сложно.

На Stepik запустили годный курс по «Troubleshooting Docker и Kubernetes: поиск и устранение проблем» В программе только важны
На Stepik запустили годный курс по «Troubleshooting Docker и Kubernetes: поиск и устранение проблем» В программе только важные аспекты: — troubleshooting Docker и образов — диагностика сетевых проблем — настройка readiness/liveness probes — отладка pod’ов, деплоев и ingress — анализ логов контейнеров и кластера — разбор ошибок CrashLoopBackOff, OOMKilled, ImagePullBackOff и других Собеседования на DevOps/SRE сейчас всё чаще строятся вокруг реальных инцидентов. Данный курс фокусируется именно на таких сценариях и помогает в подготовке к практическим вопросам 48 часов доступен со скидкой 25% ↗️ Пройти курс на Stepik

☕️ Возвращение пустых коллекций вместо null Возвращение пустых коллекций вместо null-это рекомендуемый подход для методов, возвращающих коллекции. Он упрощает обработку данных, предотвращает ошибки и делает код более предсказуемым. Преимущества 1. Улучшение читаемости: Код становится проще, так как не нужно проверять результат на null. 2. Предотвращение NullPointerException: Исключается вероятность ошибок, связанных с доступом к null. 3.Соответствие принципу наименьшего удивления: Методы всегда возвращают коллекцию, даже если она пуста. 4. Эффективность: Пустые коллекции создаются один раз и переиспользуются благодаря реализации через паттерн Singleton. 5. Совместимость с функциональным программированием: Пустые коллекции легко интегрируются в стримы и другие функциональные конструкции. Пример:

public class CacheService {
    private final Map<String, List<Object>> cache = new ConcurrentHashMap<>();

    public List<Object> getCachedValues(String key) {
        return cache.getOrDefault(key, Collections.emptyList());
    }

    public void addToCache(String key, Object value) {
        cache.computeIfAbsent(key, k -> new ArrayList<>()).add(value);
    }
}
❗️ Метод Collections.emptyList() является частью стандартной библиотеки Java (java.util.Collections). Он реализован через паттерн Singleton для повышения эффективности и безопасности. Использование пустых коллекций вместо null делает код более устойчивым и предсказуемым. Это особенно важно в сложных многопоточных приложениях или системах с большим количеством взаимосвязанных компонентов.

⌨️ Неограниченным wildcard <?> В Java <?> называется неограниченным wildcard. Он обозначает, что параметр типа может быть любым, то есть неизвестным на этапе компиляции. Это удобно, когда метод или класс работают с обобщёнными типами, но конкретный тип элемента не важен. Например, выражение List<?> означает список, элементы которого могут быть любого типа. Такой подход позволяет писать более универсальный и гибкий код, но при этом накладывает ограничения: мы не можем добавлять новые элементы в коллекцию, поскольку компилятор не знает конкретный тип элементов, чтобы обеспечить безопасность типов. ✔️ Прием метода, работающего с любым типом списка

import java.util.List;

public class WildcardExample {
    public static void printList(List<?> list) {
        for (Object element : list) { // Элементы можно читать как Object
            System.out.println(element);
        }
    }

    public static void main(String[] args) {
        List<Integer> intList = List.of(1, 2, 3);
        List<String> strList = List.of("A", "B", "C");

        printList(intList); // Вывод: 1 2 3
        printList(strList); // Вывод: A B C
    }
}
Здесь List<?> позволяет передавать любой тип списка, но мы можем безопасно читать только как Object. Ограничение на добавление элементов

public static void addElement(List<?> list) {
    // list.add("Hello"); // Ошибка компиляции!
}
#java #wildcard