🤔 Что будет, если добавлять элементы в TreeSet по возрастанию?
TreeSet всегда отсортирован по естественному порядку или Comparator. Добавление по возрастанию не вызывает проблем, структура автоматически сбалансируется (использует Red-Black Tree).
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
TreeSet всегда отсортирован по естественному порядку или Comparator. Добавление по возрастанию не вызывает проблем, структура автоматически сбалансируется (использует Red-Black Tree).
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
👍7
🤔 Как бы выразил полиморфизм в коде?
Это способность объектов разных классов реагировать на одинаковые методы по-разному. В Java полиморфизм достигается через наследование, переопределение методов и использование абстрактных классов или интерфейсов.
🚩Пример полиморфизма в коде
Наследование и переопределение методов
Использование интерфейсов
Реальный пример использования полиморфизма
Результат
Ставь 👍 и забирай 📚 Базу знаний
Это способность объектов разных классов реагировать на одинаковые методы по-разному. В Java полиморфизм достигается через наследование, переопределение методов и использование абстрактных классов или интерфейсов.
🚩Пример полиморфизма в коде
Наследование и переопределение методов
class Animal {
public void sound() {
System.out.println("Animal makes a sound");
}
}
class Dog extends Animal {
@Override
public void sound() {
System.out.println("Dog barks");
}
}
class Cat extends Animal {
@Override
public void sound() {
System.out.println("Cat meows");
}
}
public class Main {
public static void main(String[] args) {
Animal myDog = new Dog(); // Полиморфизм
Animal myCat = new Cat(); // Полиморфизм
myDog.sound(); // Вывод: Dog barks
myCat.sound(); // Вывод: Cat meows
}
}Использование интерфейсов
interface Shape {
void draw();
}
class Circle implements Shape {
@Override
public void draw() {
System.out.println("Drawing a Circle");
}
}
class Rectangle implements Shape {
@Override
public void draw() {
System.out.println("Drawing a Rectangle");
}
}
public class Main {
public static void main(String[] args) {
Shape shape1 = new Circle(); // Полиморфизм
Shape shape2 = new Rectangle(); // Полиморфизм
shape1.draw(); // Вывод: Drawing a Circle
shape2.draw(); // Вывод: Drawing a Rectangle
}
}Реальный пример использования полиморфизма
class Animal {
public void sound() {
System.out.println("Some generic animal sound");
}
}
class Dog extends Animal {
@Override
public void sound() {
System.out.println("Woof Woof");
}
}
class Cat extends Animal {
@Override
public void sound() {
System.out.println("Meow");
}
}
public class Main {
public static void main(String[] args) {
Animal[] animals = {new Dog(), new Cat(), new Animal()};
for (Animal animal : animals) {
animal.sound(); // Полиморфный вызов
}
}
}Результат
Woof Woof
Meow
Some generic animal sound
Ставь 👍 и забирай 📚 Базу знаний
👍5
🤔 В каких случаях можно использовать локальное состояние, а в каких глобальный state?
- Локальное состояние (data, ref) — когда данные:
- не нужны другим компонентам;
- касаются только внутренней логики UI (модалки, чекбоксы и т.д.).
- Глобальный state (Vuex, Pinia) — когда:
- нужно делиться данными между компонентами;
- сохраняется авторизация, корзина, настройки.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
- Локальное состояние (data, ref) — когда данные:
- не нужны другим компонентам;
- касаются только внутренней логики UI (модалки, чекбоксы и т.д.).
- Глобальный state (Vuex, Pinia) — когда:
- нужно делиться данными между компонентами;
- сохраняется авторизация, корзина, настройки.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
💊19👍1
🤔 Есть ли возможность создать два Singleton'а в Spring'е
В Spring действительно можно создать два синглтона, но тут важно понимать, что Spring использует паттерн Singleton на уровне контейнера, а не на уровне JVM. Давайте разберёмся, как это возможно.
🟠Определение Singleton в Spring
В Spring синглтон означает, что в рамках одного контекста (
🟠Два Singleton'а в Spring? Как это возможно?
Есть несколько способов создать два синглтона.
Создание двух контекстов (
Spring гарантирует синглтонность только в одном контексте. Если создать два контекста, можно получить два независимых синглтона.
Определение двух бинов одного типа
Можно создать два разных бина, просто давая им разные имена в конфигурации.
Теперь оба метода создадут разные объекты.
Изменение области (
Если изменить область (
Ставь 👍 и забирай 📚 Базу знаний
В Spring действительно можно создать два синглтона, но тут важно понимать, что Spring использует паттерн Singleton на уровне контейнера, а не на уровне JVM. Давайте разберёмся, как это возможно.
🟠Определение Singleton в Spring
В Spring синглтон означает, что в рамках одного контекста (
ApplicationContext) создаётся один экземпляр бина. @Component
public class MySingletonBean {
public MySingletonBean() {
System.out.println("Создан экземпляр MySingletonBean");
}
}
🟠Два Singleton'а в Spring? Как это возможно?
Есть несколько способов создать два синглтона.
Создание двух контекстов (
ApplicationContext) Spring гарантирует синглтонность только в одном контексте. Если создать два контекста, можно получить два независимых синглтона.
AnnotationConfigApplicationContext context1 = new AnnotationConfigApplicationContext(AppConfig.class);
AnnotationConfigApplicationContext context2 = new AnnotationConfigApplicationContext(AppConfig.class);
MySingletonBean bean1 = context1.getBean(MySingletonBean.class);
MySingletonBean bean2 = context2.getBean(MySingletonBean.class);
System.out.println(bean1 == bean2); // false! Два разных объекта
Определение двух бинов одного типа
Можно создать два разных бина, просто давая им разные имена в конфигурации.
@Configuration
public class AppConfig {
@Bean
public MySingletonBean firstSingleton() {
return new MySingletonBean();
}
@Bean
public MySingletonBean secondSingleton() {
return new MySingletonBean();
}
}
Теперь оба метода создадут разные объекты.
ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
MySingletonBean bean1 = context.getBean("firstSingleton", MySingletonBean.class);
MySingletonBean bean2 = context.getBean("secondSingleton", MySingletonBean.class);
System.out.println(bean1 == bean2); // false
Изменение области (
scope) бина Если изменить область (
@Scope), можно получить два экземпляра в одном контексте, но тогда это уже не будет синглтон. @Component
@Scope("prototype")
public class MyBean {
public MyBean() {
System.out.println("Создан новый экземпляр MyBean");
}
}
Ставь 👍 и забирай 📚 Базу знаний
👍5
🤔 Что такое ZonedDateTime?
Класс, представляющий дату, время и часовой пояс. Используется, когда важна привязка ко времени по региону (например, Europe/Moscow).
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Класс, представляющий дату, время и часовой пояс. Используется, когда важна привязка ко времени по региону (например, Europe/Moscow).
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
👍12🔥2
🤔 Что за исключение Interrupted Exception?
🚩Почему `InterruptedException` существует?
В многопоточной среде иногда требуется прервать выполнение потока, например, для завершения долгой задачи или для корректной остановки программы. Вместо грубого завершения потока (
🚩Когда выбрасывается `InterruptedException`?
Это исключение выбрасывается, если поток:
1. Ожидает (
2. Спит (
3. Блокируется на очереди (
И при этом его прерывают с помощью метода
Пример кода: обработка
Вывод программы
🚩Что делать после `InterruptedException`?
1. Просто обработать исключение (как в примере выше).
2. Передать исключение дальше — например, если метод не может корректно обработать прерывание, он передает его вызывающему коду:
🟠Восстановить флаг прерывания
если прерывание важно для логики программы:
Ставь 👍 и забирай 📚 Базу знаний
InterruptedException — это проверяемое исключение (checked exception) в Java, которое выбрасывается, когда поток (Thread) прерывается во время выполнения метода, который поддерживает прерывание (например, sleep(), wait(), join() и др.). 🚩Почему `InterruptedException` существует?
В многопоточной среде иногда требуется прервать выполнение потока, например, для завершения долгой задачи или для корректной остановки программы. Вместо грубого завершения потока (
Thread.stop(), который устарел и считается небезопасным) Java предлагает мягкий способ прерывания через флаг прерывания и обработку InterruptedException.🚩Когда выбрасывается `InterruptedException`?
Это исключение выбрасывается, если поток:
1. Ожидает (
wait(), join()) 2. Спит (
sleep()) 3. Блокируется на очереди (
BlockingQueue.take(), LockSupport.park()) И при этом его прерывают с помощью метода
interrupt().Пример кода: обработка
InterruptedException class MyThread extends Thread {
public void run() {
try {
System.out.println("Поток засыпает...");
Thread.sleep(5000); // Поток засыпает на 5 секунд
} catch (InterruptedException e) {
System.out.println("Поток был прерван во время сна!");
}
System.out.println("Поток продолжает работу...");
}
}
public class Main {
public static void main(String[] args) {
MyThread t = new MyThread();
t.start();
try {
Thread.sleep(2000); // Даем потоку 2 секунды поработать
} catch (InterruptedException e) {
e.printStackTrace();
}
t.interrupt(); // Прерываем поток
}
}Вывод программы
Поток засыпает...
Поток был прерван во время сна!
Поток продолжает работу...
🚩Что делать после `InterruptedException`?
1. Просто обработать исключение (как в примере выше).
2. Передать исключение дальше — например, если метод не может корректно обработать прерывание, он передает его вызывающему коду:
void myMethod() throws InterruptedException {
Thread.sleep(1000); // Может выбросить исключение
}🟠Восстановить флаг прерывания
если прерывание важно для логики программы:
catch (InterruptedException e) {
Thread.currentThread().interrupt(); // Восстанавливаем флаг прерывания
}Ставь 👍 и забирай 📚 Базу знаний
👍8
🤔 Какой класс реализует динамический массив в Java?
ArrayList — это реализация динамического массива в Java, основанная на обычном массиве. Он автоматически расширяется при добавлении новых элементов и предоставляет методы для управления коллекцией — добавления, удаления, сортировки и доступа по индексу.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
ArrayList — это реализация динамического массива в Java, основанная на обычном массиве. Он автоматически расширяется при добавлении новых элементов и предоставляет методы для управления коллекцией — добавления, удаления, сортировки и доступа по индексу.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
👍11💊1
🤔 Какие минусы Rest в высоконагруженных сис-мах?
REST – это удобный и популярный архитектурный стиль, но в высоконагруженных системах у него есть недостатки.
🚩Высокие накладные расходы HTTP (избыточность)
Проблема: Каждый REST-запрос передаёт много мета-информации (заголовки, cookies, CORS, JSON/XML).
В WebSocket/gRPC запрос выглядел бы так
🚩REST-запросы неэффективны для чатов и real-time
Проблема: REST использует "запрос-ответ", но **не поддерживает real-time.
Что происходит в REST?
1. Клиент делает HTTP-запрос к серверу.
2. Сервер обрабатывает и отправляет ответ.
3. Если клиенту нужно новое сообщение, он снова делает запрос (Polling) или использует Long Polling.
🚩Проблемы с кэшированием
Проблема: Не все REST-запросы можно кэшировать.
Кэшируются только
🚩REST API неэффективен для сложных запросов
Проблема: REST требует много отдельных запросов, если нужно загрузить связанные данные.
Пример проблемы REST:
Получить пользователя:
Получить заказы этого пользователя:
Получить детали заказов:
Ставь 👍 и забирай 📚 Базу знаний
REST – это удобный и популярный архитектурный стиль, но в высоконагруженных системах у него есть недостатки.
🚩Высокие накладные расходы HTTP (избыточность)
Проблема: Каждый REST-запрос передаёт много мета-информации (заголовки, cookies, CORS, JSON/XML).
GET /users/123 HTTP/1.1
Host: api.example.com
Authorization: Bearer token123
Content-Type: application/json
В WebSocket/gRPC запрос выглядел бы так
{ "userId": 123 }🚩REST-запросы неэффективны для чатов и real-time
Проблема: REST использует "запрос-ответ", но **не поддерживает real-time.
Что происходит в REST?
1. Клиент делает HTTP-запрос к серверу.
2. Сервер обрабатывает и отправляет ответ.
3. Если клиенту нужно новое сообщение, он снова делает запрос (Polling) или использует Long Polling.
🚩Проблемы с кэшированием
Проблема: Не все REST-запросы можно кэшировать.
Кэшируются только
GET-запросы (если сервер поддерживает ETag и Cache-Control). POST, PUT, DELETE – не кэшируются, потому что меняют данные. 🚩REST API неэффективен для сложных запросов
Проблема: REST требует много отдельных запросов, если нужно загрузить связанные данные.
Пример проблемы REST:
Получить пользователя:
/users/123 Получить заказы этого пользователя:
/users/123/orders Получить детали заказов:
/orders/456/detailsСтавь 👍 и забирай 📚 Базу знаний
👍10
🤔 Что означают отношения "являюсь" и "имею" в ООП?
- "Являюсь" (is-a) — отношение наследования:
- Кошка is-a Животное → class Cat extends Animal.
- "Имею" (has-a) — отношение композиции/агрегации:
- Машина has-a Двигатель → class Car { Engine engine; }.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
- "Являюсь" (is-a) — отношение наследования:
- Кошка is-a Животное → class Cat extends Animal.
- "Имею" (has-a) — отношение композиции/агрегации:
- Машина has-a Двигатель → class Car { Engine engine; }.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
👍9🔥3
🤔 Является ли коллекция HashMap потоком безопасной?
Коллекция
🟠
Оборачивает ее (или любую другую карту) в потокобезопасную обёртку, гарантируя безопасность при доступе из разных потоков. Однако при использовании этого метода важно помнить, что если итерация по коллекции происходит в многопоточной среде, необходимо синхронизировать весь блок итерации на возвращённой карте для предотвращения конкурентных модификаций.
🟠
Предоставляет потокобезопасную реализацию карты без блокировки всей карты.
Ставь 👍 и забирай 📚 Базу знаний
Коллекция
HashMap не является потокобезопасной. Это означает, что при одновременном доступе к нему из нескольких потоков без должной синхронизации могут возникнуть проблемы, такие как потеря данных, гонки за данные и другие виды состояний гонки. Если один поток изменяет ее структуру (например, добавляя или удаляя элементы), в то время как другой поток итерирует по ней или также пытается внести изменения, результаты могут быть непредсказуемыми.🟠
Collections.synchronizedMap(Map)Оборачивает ее (или любую другую карту) в потокобезопасную обёртку, гарантируя безопасность при доступе из разных потоков. Однако при использовании этого метода важно помнить, что если итерация по коллекции происходит в многопоточной среде, необходимо синхронизировать весь блок итерации на возвращённой карте для предотвращения конкурентных модификаций.
Map<String, String> map = Collections.synchronizedMap(new HashMap<>());
🟠
ConcurrentHashMapПредоставляет потокобезопасную реализацию карты без блокировки всей карты.
ConcurrentHashMap разработан для высокой конкуренции и эффективности при доступе из множества потоков, обеспечивая лучшую производительность по сравнению с synchronizedMap. ConcurrentHashMap позволяет одновременно читать данные из карты несколькими потоками без блокировки и записывать данные при минимальной блокировке.Map<String, String> concurrentMap = new ConcurrentHashMap<>();
Ставь 👍 и забирай 📚 Базу знаний
👍10
🤔 Что означает LoadFactor?
LoadFactor — это порог заполнения хеш-таблицы, при котором происходит расширение (rehash).
По умолчанию 0.75, то есть при заполнении 75% бакетов таблица увеличивается вдвое, чтобы избежать коллизий и деградации производительности.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
LoadFactor — это порог заполнения хеш-таблицы, при котором происходит расширение (rehash).
По умолчанию 0.75, то есть при заполнении 75% бакетов таблица увеличивается вдвое, чтобы избежать коллизий и деградации производительности.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
👍6🔥4
🤔 Что знаешь про поверхностное копирование?
Поверхностное копирование (Shallow Copy) – это процесс создания нового объекта, который содержит ссылки на те же вложенные объекты, что и оригинал.
🟠Как сделать поверхностное копирование в Java?
Способ 1: Метод
🟠Проблема с вложенными объектами (общие ссылки)
Если объект содержит вложенные объекты, они не копируются, а передаются по ссылке.
🟠Как сделать глубокую копию? (Deep Copy)
Решение: Создать новый вложенный объект в
Ставь 👍 и забирай 📚 Базу знаний
Поверхностное копирование (Shallow Copy) – это процесс создания нового объекта, который содержит ссылки на те же вложенные объекты, что и оригинал.
🟠Как сделать поверхностное копирование в Java?
Способ 1: Метод
clone() (реализация Cloneable) class Person implements Cloneable {
String name;
public Person(String name) {
this.name = name;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone(); // Поверхностное копирование
}
}
public class Main {
public static void main(String[] args) throws CloneNotSupportedException {
Person original = new Person("Иван");
Person copy = (Person) original.clone();
System.out.println(copy.name); // Иван
}
}🟠Проблема с вложенными объектами (общие ссылки)
Если объект содержит вложенные объекты, они не копируются, а передаются по ссылке.
class Address {
String city;
public Address(String city) { this.city = city; }
}
class User implements Cloneable {
String name;
Address address;
public User(String name, Address address) {
this.name = name;
this.address = address;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone(); // Поверхностное копирование
}
}
public class Main {
public static void main(String[] args) throws CloneNotSupportedException {
Address address = new Address("Москва");
User original = new User("Иван", address);
User copy = (User) original.clone();
copy.address.city = "Санкт-Петербург"; // Меняем адрес у копии
System.out.println(original.address.city); // Санкт-Петербург (изменилось и у оригинала!)
}
}🟠Как сделать глубокую копию? (Deep Copy)
Решение: Создать новый вложенный объект в
clone() @Override
protected Object clone() throws CloneNotSupportedException {
User clonedUser = (User) super.clone();
clonedUser.address = new Address(this.address.city); // Копируем вложенный объект
return clonedUser;
}
Ставь 👍 и забирай 📚 Базу знаний
👍7
🤔 Как осуществить быстрый поиск без сортировки int массива на миллионы значений?
- использовать хеш-таблицу (если нужна проверка наличия);
- применять бинарный поиск, если данные отсортированы;
- если нет — использовать Set, Map, или хешированную структуру для O(1) поиска.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
- использовать хеш-таблицу (если нужна проверка наличия);
- применять бинарный поиск, если данные отсортированы;
- если нет — использовать Set, Map, или хешированную структуру для O(1) поиска.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
💊15
🤔 В чем разница JPQL и HQL?
JPQL (Java Persistence Query Language) и HQL (Hibernate Query Language) – это языки запросов, похожие на SQL, но работающие с объектами, а не с таблицами БД.
Главное отличие: JPQL – стандарт JPA, а HQL – специфичен для Hibernate.
🚩JPQL (Java Persistence Query Language)
Стандартный язык JPA, работает со всеми JPA-совместимыми провайдерами (
Опирается на JPA Entity-классы, а не на реальные таблицы в БД.
🚩HQL (Hibernate Query Language)
Язык запросов специфичный для Hibernate.
Работает аналогично JPQL, но поддерживает дополнительные возможности, например, вызов методов и фильтрацию по подзапросам.
Ставь 👍 и забирай 📚 Базу знаний
JPQL (Java Persistence Query Language) и HQL (Hibernate Query Language) – это языки запросов, похожие на SQL, но работающие с объектами, а не с таблицами БД.
Главное отличие: JPQL – стандарт JPA, а HQL – специфичен для Hibernate.
🚩JPQL (Java Persistence Query Language)
Стандартный язык JPA, работает со всеми JPA-совместимыми провайдерами (
Hibernate, EclipseLink, OpenJPA). Опирается на JPA Entity-классы, а не на реальные таблицы в БД.
TypedQuery<User> query = entityManager.createQuery(
"SELECT u FROM User u WHERE u.age > :age", User.class);
query.setParameter("age", 18);
List<User> users = query.getResultList();
🚩HQL (Hibernate Query Language)
Язык запросов специфичный для Hibernate.
Работает аналогично JPQL, но поддерживает дополнительные возможности, например, вызов методов и фильтрацию по подзапросам.
Query query = session.createQuery(
"FROM User u WHERE LENGTH(u.name) > 5"); // Использование функции LENGTH()
List<User> users = query.list();
Ставь 👍 и забирай 📚 Базу знаний
👍7
🤔 Для чего используется WeakHashMap?
Хранит ключи через слабые ссылки, чтобы:
- позволить сборку мусора, если на ключ больше нет сильных ссылок;
- удобно при кэшировании, где важен автоматический сброс.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
Хранит ключи через слабые ссылки, чтобы:
- позволить сборку мусора, если на ключ больше нет сильных ссылок;
- удобно при кэшировании, где важен автоматический сброс.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
🔥7👍4
🤔 Что знаешь об интерфейсе FileFilter?
🚩Как работает `FileFilter`?
Этот интерфейс содержит всего один метод:
🚩Пример использования `FileFilter`
Допустим, мы хотим отфильтровать все файлы
Выходные данные (если в папке
🚩Lambda-версия (Java 8+)
Вместо анонимного класса можно использовать лямбда-выражение:
🚩Разница между `FileFilter` и `FilenameFilter`
Ставь 👍 и забирай 📚 Базу знаний
FileFilter — это интерфейс в пакете java.io, используемый для фильтрации файлов в каталогах. Он применяется в методе listFiles(FileFilter filter) класса File и позволяет выбрать только те файлы, которые соответствуют заданным критериям.🚩Как работает `FileFilter`?
Этот интерфейс содержит всего один метод:
boolean accept(File pathname);
🚩Пример использования `FileFilter`
Допустим, мы хотим отфильтровать все файлы
.txt в заданной папке:import java.io.File;
import java.io.FileFilter;
public class TxtFileFilterExample {
public static void main(String[] args) {
File directory = new File("C:/example"); // Укажите свою папку
// Используем FileFilter для выбора файлов с расширением .txt
FileFilter txtFilter = new FileFilter() {
@Override
public boolean accept(File file) {
return file.isFile() && file.getName().endsWith(".txt");
}
};
// Получаем список файлов, соответствующих фильтру
File[] txtFiles = directory.listFiles(txtFilter);
// Выводим найденные файлы
if (txtFiles != null) {
for (File file : txtFiles) {
System.out.println("Файл: " + file.getName());
}
}
}
}
Выходные данные (если в папке
C:/example есть файлы .txt)Файл: notes.txt
Файл: tasks.txt
🚩Lambda-версия (Java 8+)
Вместо анонимного класса можно использовать лямбда-выражение:
FileFilter txtFilter = file -> file.isFile() && file.getName().endsWith(".txt");🚩Разница между `FileFilter` и `FilenameFilter`
FileFilter принимает объект File, позволяя фильтровать как файлы, так и каталоги.FilenameFilter принимает только имя файла (без пути).import java.io.File;
import java.io.FilenameFilter;
public class TxtFilenameFilterExample {
public static void main(String[] args) {
File directory = new File("C:/example");
// Фильтр для файлов .txt
FilenameFilter txtFilter = (dir, name) -> name.endsWith(".txt");
String[] txtFiles = directory.list(txtFilter);
if (txtFiles != null) {
for (String file : txtFiles) {
System.out.println("Файл: " + file);
}
}
}
}
Ставь 👍 и забирай 📚 Базу знаний
👍7
🤔 Что такое mock в тестировании?
Это имитация объекта для тестирования.
1. Используется, чтобы изолировать тестируемый код от зависимостей.
2. Помогает проверить поведение системы в различных сценариях без реального взаимодействия.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
1. Используется, чтобы изолировать тестируемый код от зависимостей.
2. Помогает проверить поведение системы в различных сценариях без реального взаимодействия.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
👍3
🤔 В чем разница между Statement и PreparedStatement?
Это два интерфейса в JDBC, которые используются для выполнения SQL-запросов к базе данных. Основные различия между ними касаются производительности, безопасности и удобства использования.
🚩Statement
Statement используется для выполнения простых SQL-запросов. Запросы формируются и передаются в базу данных в виде строки. Подходит для простых запросов. Не оптимизирован для повторного выполнения. Каждый раз, когда запрос передается в базу данных, он компилируется и выполняется заново. Уязвим к SQL-инъекциям. Поскольку запрос формируется путем конкатенации строк, злоумышленники могут внедрять вредоносный SQL-код.
🟠PreparedStatement
PreparedStatement используется для выполнения предварительно скомпилированных SQL-запросов. Это позволяет повысить производительность и безопасность. Оптимизирован для повторного выполнения. Запрос компилируется только один раз, а затем может многократно выполняться с разными параметрами, что повышает производительность. Защита от SQL-инъекций. Использует параметризованные запросы, которые помогают избежать уязвимостей, связанных с SQL-инъекциями. Удобство работы с параметрами. Позволяет устанавливать значения параметров с использованием методов setInt(), setString() и других.
🚩Основные различия
🟠Производительность
🟠Безопасность
🟠Удобство использования
🟠Типы запросов
Оба интерфейса могут выполнять запросы типа
Ставь 👍 и забирай 📚 Базу знаний
Это два интерфейса в JDBC, которые используются для выполнения SQL-запросов к базе данных. Основные различия между ними касаются производительности, безопасности и удобства использования.
🚩Statement
Statement используется для выполнения простых SQL-запросов. Запросы формируются и передаются в базу данных в виде строки. Подходит для простых запросов. Не оптимизирован для повторного выполнения. Каждый раз, когда запрос передается в базу данных, он компилируется и выполняется заново. Уязвим к SQL-инъекциям. Поскольку запрос формируется путем конкатенации строк, злоумышленники могут внедрять вредоносный SQL-код.
Statement stmt = connection.createStatement();
String query = "SELECT * FROM employees WHERE department = 'HR'";
ResultSet rs = stmt.executeQuery(query);
while (rs.next()) {
int id = rs.getInt("id");
String name = rs.getString("name");
System.out.println("ID: " + id + ", Name: " + name);
}
stmt.close();
🟠PreparedStatement
PreparedStatement используется для выполнения предварительно скомпилированных SQL-запросов. Это позволяет повысить производительность и безопасность. Оптимизирован для повторного выполнения. Запрос компилируется только один раз, а затем может многократно выполняться с разными параметрами, что повышает производительность. Защита от SQL-инъекций. Использует параметризованные запросы, которые помогают избежать уязвимостей, связанных с SQL-инъекциями. Удобство работы с параметрами. Позволяет устанавливать значения параметров с использованием методов setInt(), setString() и других.
String query = "SELECT * FROM employees WHERE department = ?";
PreparedStatement pstmt = connection.prepareStatement(query);
pstmt.setString(1, "HR");
ResultSet rs = pstmt.executeQuery();
while (rs.next()) {
int id = rs.getInt("id");
String name = rs.getString("name");
System.out.println("ID: " + id + ", Name: " + name);
}
pstmt.close();
🚩Основные различия
🟠Производительность
Statement: Каждый запрос компилируется заново, что снижает производительность при многократном выполнении одного и того же запроса.PreparedStatement: Запрос компилируется один раз и может многократно выполняться с разными параметрами, что повышает производительность.🟠Безопасность
Statement: Уязвим к SQL-инъекциям, поскольку запросы формируются путем конкатенации строк.PreparedStatement: Использует параметризованные запросы, что защищает от SQL-инъекций.🟠Удобство использования
Statement: Подходит для простых, одноразовых запросов.PreparedStatement: Удобен для многократного выполнения запросов с разными параметрами.🟠Типы запросов
Оба интерфейса могут выполнять запросы типа
SELECT, INSERT, UPDATE, DELETE, но PreparedStatement более удобен для запросов с параметрами.Ставь 👍 и забирай 📚 Базу знаний
👍11
🤔 Какие есть механизмы для сложных запросов с множеством параметров?
В Spring и Hibernate можно использовать:
- JPQL — объектно-ориентированный SQL.
- Native SQL — для сложных или нестандартных запросов.
- Criteria API — программно-строимые типобезопасные запросы.
- Specification — шаблон для построения динамических запросов с множеством условий.
- @Query — аннотация в Spring Data для написания запросов прямо в интерфейсе репозитория.
- QueryDSL — внешняя библиотека, генерирующая типобезопасные запросы.
Все эти подходы позволяют создавать гибкие, читаемые и поддерживаемые SQL-запросы даже с десятками параметров и условий.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
- JPQL — объектно-ориентированный SQL.
- Native SQL — для сложных или нестандартных запросов.
- Criteria API — программно-строимые типобезопасные запросы.
- Specification — шаблон для построения динамических запросов с множеством условий.
-
- QueryDSL — внешняя библиотека, генерирующая типобезопасные запросы.
Все эти подходы позволяют создавать гибкие, читаемые и поддерживаемые SQL-запросы даже с десятками параметров и условий.
Ставь 👍 если знал ответ, 🔥 если нет
Забирай 📚 Базу знаний
👍6
🤔 Слышал ли что то про метод intern?
Метод
Он добавляет строку в String Pool и возвращает её ссылку, если строка уже там есть.
🚩Как работает `intern()`?
Без
С
🚩Что такое `String Pool`?
Это специальная область памяти, где хранятся уникальные строковые литералы.
Все строковые литералы (
🚩Когда использовать `intern()`?
Когда у вас много одинаковых строк в памяти (например, имена, идентификаторы).
Для экономии памяти, если строки часто дублируются.
В парсинге JSON, XML – одни и те же строки могут повторяться тысячи раз.
Ставь 👍 и забирай 📚 Базу знаний
Метод
intern() в классе String используется для оптимизации памяти. Он добавляет строку в String Pool и возвращает её ссылку, если строка уже там есть.
🚩Как работает `intern()`?
Без
intern() – строки создаются в Heap (куче) String s1 = new String("Hello"); // В куче (Heap)
String s2 = new String("Hello");
System.out.println(s1 == s2); // false (разные объекты)С
intern() – строки хранятся в String Pool String s1 = new String("Hello").intern();
String s2 = new String("Hello").intern();
System.out.println(s1 == s2); // true (одна строка в String Pool)🚩Что такое `String Pool`?
Это специальная область памяти, где хранятся уникальные строковые литералы.
Все строковые литералы (
"Hello") по умолчанию хранятся в String Pool. String s1 = "Hello"; // В String Pool
String s2 = "Hello"; // Ссылается на тот же объект
System.out.println(s1 == s2); // true
🚩Когда использовать `intern()`?
Когда у вас много одинаковых строк в памяти (например, имена, идентификаторы).
Для экономии памяти, если строки часто дублируются.
В парсинге JSON, XML – одни и те же строки могут повторяться тысячи раз.
List<String> names = new ArrayList<>();
for (int i = 0; i < 100000; i++) {
names.add(("User" + (i % 100)).intern()); // Используем String Pool
}
Ставь 👍 и забирай 📚 Базу знаний
👍6🔥1