Java for Beginner
675 subscribers
548 photos
155 videos
12 files
839 links
Канал от новичков для новичков!
Изучайте Java вместе с нами!
Здесь мы обмениваемся опытом и постоянно изучаем что-то новое!

Наш YouTube канал - https://www.youtube.com/@Java_Beginner-Dev

Наш канал на RUTube - https://rutube.ru/channel/37896292/
Download Telegram
Всем доброго субботнего утра!☝️

Какие планы на выходные?

А давайте делится идеями пет проектов, давайте посмотрим кто заморочился больше?

Желаю каждому участнику канала, прекрасных выходных!🍻

Завтра увидимся на лайф-кодинге!😜
This media is not supported in your browser
VIEW IN TELEGRAM
Доброе утро!

Напоминаю, что сегодня в 16:00 по МСК, мы вновь встречаемся, чтобы разобрать структурные паттерны проектирования.

Запасайтесь хорошим настроением и печеньками, ждем всех!

🖐
И вот зачем он это сделал? 😡😂

https://t.me/Java_for_beginner_dev

#Mems
Запись нашей сегодняшней встречи -
https://youtu.be/_jnIaJz_mJc
https://rutube.ru/video/506c291cd9b61894b4c99ac8c168e115/

Огромный респект тем кто смог прийти, за участие и подсказки!
@Alexander_Gors отдельное спасибо, за лайф-кодинг!)

Сегодня, мы совместно рассмотрели на примерах структурные паттерны проектирования:
- Адаптер (Adapter)
- Мост(Bridge)
- Компоновщик (Composite)
- Декоратор (Decorator, Wrapper)
- Фасад (Facade)
- Легковес (Flyweight)
- Заместитель (Proxy)


Смотрите, комментируйте, задавайте вопросы! Обязательно подписывайтесь на ютуб и рутюб каналы!!!

Гит репозиторий с результатом - https://github.com/Oleborn/PatternsOfCoding

Всем теплой ночи и легкой рабочей недели! 🫡✌️
Встреча_в_Телемосте_11_08_24_20_03_54_—_запись.webm
227 MB
Для тех кто не смог победить блокировку Youtube и не желает заходить в Rutube выкладываю видео тут!

Смотрите на здоровье) 🫡

Но с вас подписка на Youtube и Rutube и делитесь этими видео, буду крайне благодарен!)))

#online_meeting
ConcurrentLinkedQueue, внутреннее устройство, особенности и преимущества

ConcurrentLinkedQueue — это неконкурирующая неблокирующая очередь на основе связанного списка, предоставляемая в пакете java.util.concurrent. Она поддерживает высокую степень конкурентности благодаря использованию алгоритмов без блокировок (lock-free algorithms).

Особенности и преимущества

Неконкурирующая: Методы ConcurrentLinkedQueue реализованы таким образом, что могут использоваться несколькими потоками одновременно без необходимости явной синхронизации.
Неблокирующая: Основана на алгоритмах без блокировок, что позволяет избежать блокировок и взаимных блокировок, повышая производительность в условиях высокой конкурентности.
Поддержка FIFO (First-In-First-Out): Гарантирует, что элементы будут обрабатываться в порядке их добавления.


Внутреннее устройство

ConcurrentLinkedQueue использует алгоритм Майкла-скотта (Michael-Scott) для реализации неблокирующей очереди на основе связанного списка. Основные компоненты включают узлы (nodes) и ссылки на голову (head) и хвост (tail) очереди.

Внутренний класс Node:

Узел содержит элемент данных и ссылку на следующий узел.
private static class Node<E> {
volatile E item;
volatile Node<E> next;

Node(E item) {
this.item = item;
}
}


Голова и хвост очереди:

Очередь хранит ссылки на голову и хвост, которые используются для вставки и удаления элементов.
private transient volatile Node<E> head;
private transient volatile Node<E> tail;


Алгоритм Майкла-скотта:

Этот алгоритм основан на атомарных операциях, таких как compareAndSet, для безопасного обновления ссылок на узлы без использования блокировок.
public boolean offer(E e) {
final Node<E> newNode = new Node<>(e);
for (Node<E> t = tail, p = t; ; ) {
Node<E> q = p.next;
if (q == null) {
if (p.next = q) {
tail = newNode;
return true;
}
} else {
p = (t != tail) ? tail : q;
}
}
}


Преимущества использования

Высокая производительность в многопоточных средах: Благодаря отсутствию блокировок, ConcurrentLinkedQueue показывает высокую производительность даже при большом количестве конкурирующих потоков.
Минимизация блокировок и ожиданий: Использование атомарных операций позволяет избежать задержек, связанных с блокировками.
Безопасность при многопоточном доступе: Методы очереди обеспечивают безопасность при доступе из нескольких потоков без дополнительной синхронизации.

import java.util.concurrent.ConcurrentLinkedQueue;

public class ConcurrentLinkedQueueExample {
public static void main(String[] args) {
ConcurrentLinkedQueue<Integer> queue = new ConcurrentLinkedQueue<>();

// Вставка элементов
queue.offer(1);
queue.offer(2);
queue.offer(3);

// Извлечение элементов
System.out.println(queue.poll()); // 1
System.out.println(queue.poll()); // 2
System.out.println(queue.poll()); // 3
}
}


Ссылки на полезные статьи (спасибо авторам за проделанную работу) :
https://www.baeldung.com/java-queue-linkedblocking-concurrentlinked
https://www.geeksforgeeks.org/concurrentlinkedqueue-in-java-with-examples/

#Java #Training #Medium #ConcurrentLinkedQueue
Варианты ответа:
Anonymous Quiz
17%
15
17%
161
50%
18
17%
21
Что выведет код?

import java.util.concurrent.ConcurrentLinkedQueue;

public class ConcurrentLinkedQueueChallenge {
public static void main(String[] args) {
ConcurrentLinkedQueue<Integer> queue = new ConcurrentLinkedQueue<>();
for (int i = 1; i <= 5; i++) {
queue.offer(i);
}

int result = 1;
for (int i = 0; i < 3; i++) {
Integer val = queue.poll();
if (val != null) {
result *= val;
}
queue.offer(result);
}

int sum = 0;
for (Integer val : queue) {
sum += val;
}

System.out.println("Sum: " + sum);
}
}


#Tasks
Мем для IQ > 100!😱

Внимание! Возможен перелом мозга! Будьте внимательнее!!!

https://t.me/Java_for_beginner_dev

#Mems
Основные методы ConcurrentLinkedQueue и примеры использования

offer(E e):

Вставляет элемент в конец очереди. Возвращает true после успешного добавления.
ConcurrentLinkedQueue<Integer> queue = new ConcurrentLinkedQueue<>();
queue.offer(1);
queue.offer(2);


poll():

Извлекает и удаляет элемент из головы очереди. Возвращает null, если очередь пуста.
Integer item = queue.poll();
System.out.println(item); // 1


peek():

Возвращает элемент из головы очереди без его удаления. Возвращает null, если очередь пуста.

Integer item = queue.peek();
System.out.println(item); // 2


isEmpty():

Проверяет, пуста ли очередь.
boolean isEmpty = queue.isEmpty();
System.out.println(isEmpty); // false


size():

Возвращает количество элементов в очереди. Обратите внимание, что этот метод может быть дорогим, так как требует полной итерации по очереди для подсчета элементов.
int size = queue.size();
System.out.println(size); // 1


Примеры использования

Производитель-потребитель:
import java.util.concurrent.ConcurrentLinkedQueue;

public class ProducerConsumerExample {
public static void main(String[] args) {
ConcurrentLinkedQueue<Integer> queue = new ConcurrentLinkedQueue<>();

// Производитель
new Thread(() -> {
for (int i = 0; i < 10; i++) {
queue.offer(i);
System.out.println("Produced: " + i);
}
}).start();

// Потребитель
new Thread(() -> {
for (int i = 0; i < 10; i++) {
Integer item;
while ((item = queue.poll()) == null) {
// Ожидание элемента
}
System.out.println("Consumed: " + item);
}
}).start();
}
}


Очередь задач:
import java.util.concurrent.ConcurrentLinkedQueue;

public class TaskQueueExample {
public static void main(String[] args) {
ConcurrentLinkedQueue<Runnable> taskQueue = new ConcurrentLinkedQueue<>();

// Запуск потока исполнителя задач
new Thread(() -> {
while (true) {
Runnable task = taskQueue.poll();
if (task != null) {
task.run();
}
}
}).start();

// Добавление задач в очередь
taskQueue.offer(() -> System.out.println("Task 1 executed"));
taskQueue.offer(() -> System.out.println("Task 2 executed"));
}
}


#Java #Training #Medium #ConcurrentLinkedQueue
DelayQueue, внутреннее устройство и преимущества

DelayQueue — это специализированная очередь из пакета java.util.concurrent, которая предназначена для хранения элементов до истечения их задержки. Она основана на приоритетной очереди и используется в ситуациях, где необходимо реализовать задержку выполнения задач.

Внутреннее устройство DelayQueue

DelayQueue реализует интерфейсы BlockingQueue и Queue. Она состоит из элементов, которые должны реализовать интерфейс Delayed. Этот интерфейс требует реализации метода getDelay(TimeUnit unit), который возвращает оставшееся время задержки элемента в указанных единицах.

Элементы DelayQueue:

Каждый элемент в DelayQueue должен реализовать интерфейс Delayed.
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit;

class DelayedElement implements Delayed {
private final long delayTime;
private final long creationTime;

public DelayedElement(long delayTime) {
this.delayTime = delayTime;
this.creationTime = System.currentTimeMillis();
}

@Override
public long getDelay(TimeUnit unit) {
long elapsedTime = System.currentTimeMillis() - creationTime;
return unit.convert(delayTime - elapsedTime, TimeUnit.MILLISECONDS);
}

@Override
public int compareTo(Delayed o) {
if (this.getDelay(TimeUnit.MILLISECONDS) < o.getDelay(TimeUnit.MILLISECONDS)) {
return -1;
}
if (this.getDelay(TimeUnit.MILLISECONDS) > o.getDelay(TimeUnit.MILLISECONDS)) {
return 1;
}
return 0;
}
}


Внутренняя структура:

DelayQueue использует приоритетную очередь для хранения элементов. Это позволяет эффективно управлять элементами на основе их времени задержки.
Приоритетная очередь реализуется на основе кучи (heap), что обеспечивает логарифмическое время выполнения операций вставки и удаления.


Механизм блокировки:

DelayQueue использует методы ReentrantLock для обеспечения потокобезопасности. Это позволяет избежать гонок и обеспечивает корректное поведение в многопоточной среде.

private final transient ReentrantLock lock = new ReentrantLock();
private final Condition available = lock.newCondition();


Извлечение элементов:

При попытке извлечения элемента из DelayQueue, если задержка элемента еще не истекла, поток будет заблокирован до истечения задержки.
public E take() throws InterruptedException {
final ReentrantLock lock = this.lock;
lock.lockInterruptibly();
try {
for (;;) {
E first = q.peek();
if (first == null) {
available.await();
} else {
long delay = first.getDelay(TimeUnit.NANOSECONDS);
if (delay <= 0) {
return q.poll();
}
first = null;
available.awaitNanos(delay);
}
}
} finally {
lock.unlock();
}
}


Преимущества DelayQueue

Управление задержками: DelayQueue идеально подходит для задач, где необходимо управлять задержками, таких как планирование задач или кэширование с истечением времени.
Потокобезопасность: Использование механизмов блокировки и приоритетной очереди обеспечивает безопасность в многопоточных приложениях.
Гибкость: Возможность использовать любые объекты, реализующие интерфейс Delayed, позволяет легко адаптировать DelayQueue под различные задачи.


Ссылки на полезные статьи (спасибо авторам за проделанную работу) :
https://www.baeldung.com/java-delay-queue
https://habr.com/ru/articles/599603/

#Java #Training #Medium #DelayQueue
Варианты ответа:
Anonymous Quiz
0%
3
33%
4
67%
6
0%
9
Что выведет код?

import java.util.concurrent.DelayQueue;
import java.util.concurrent.Delayed;
import java.util.concurrent.TimeUnit;

class DelayedElement implements Delayed {
private final long delayTime;
private final long creationTime;

public DelayedElement(long delayTime) {
this.delayTime = delayTime;
this.creationTime = System.currentTimeMillis();
}

@Override
public long getDelay(TimeUnit unit) {
long diff = creationTime + delayTime - System.currentTimeMillis();
return unit.convert(diff, TimeUnit.MILLISECONDS);
}

@Override
public int compareTo(Delayed o) {
return Long.compare(this.getDelay(TimeUnit.MILLISECONDS), o.getDelay(TimeUnit.MILLISECONDS));
}

@Override
public String toString() {
return "DelayedElement{" + "delayTime=" + delayTime + '}';
}
}

public class DelayQueueChallenge {
public static void main(String[] args) throws InterruptedException {
DelayQueue<DelayedElement> queue = new DelayQueue<>();
queue.offer(new DelayedElement(1000));
queue.offer(new DelayedElement(2000));
queue.offer(new DelayedElement(3000));

long startTime = System.currentTimeMillis();

int sum = 0;
for (int i = 0; i < 3; i++) {
DelayedElement element = queue.take();
long elapsedTime = (System.currentTimeMillis() - startTime) / 1000;
sum += elapsedTime;
System.out.println("Taken: " + element + " at " + elapsedTime + " seconds");
}

System.out.println("Sum: " + sum);
}
}


#Tasks
Среди вас есть кто видел его вживую?🧐

https://t.me/Java_for_beginner_dev

#Mems
Основные методы DelayQueue и примеры использования

add(E e):

Добавляет элемент в очередь. Если очередь полна, генерируется исключение.
DelayQueue<DelayedElement> delayQueue = new DelayQueue<>();
delayQueue.add(new DelayedElement(1000));


put(E e):

Добавляет элемент в очередь. Если очередь полна, операция блокируется до освобождения места.
delayQueue.put(new DelayedElement(2000));


offer(E e):

Добавляет элемент в очередь, если это возможно. Возвращает true, если элемент был добавлен, и false в противном случае.
boolean added = delayQueue.offer(new DelayedElement(3000));


take():

Извлекает и удаляет элемент из головы очереди, ожидая при необходимости до истечения задержки.
DelayedElement element = delayQueue.take();


poll(long timeout, TimeUnit unit):

Извлекает и удаляет элемент из головы очереди, ожидая до указанного времени, если необходимо.
DelayedElement element = delayQueue.poll(2, TimeUnit.SECONDS);


clear():

Удаляет все элементы из очереди.
delayQueue.clear();


remove(Object o):

Удаляет указанный элемент из очереди, если он присутствует.
delayQueue.remove(someElement);


Примеры использования

Запланированные задачи:
DelayQueue можно использовать для планирования выполнения задач через определенные промежутки времени.
import java.util.concurrent.DelayQueue;

public class ScheduledTaskExample {
public static void main(String[] args) throws InterruptedException {
DelayQueue<DelayedElement> delayQueue = new DelayQueue<>();

// Добавление задач
delayQueue.put(new DelayedElement(3000)); // 3 секунды задержки
delayQueue.put(new DelayedElement(5000)); // 5 секунд задержки

// Обработка задач
while (!delayQueue.isEmpty()) {
DelayedElement element = delayQueue.take();
System.out.println("Task executed: " + element);
}
}
}


Кэширование с истечением срока действия:

DelayQueue можно использовать для реализации кэша, элементы которого удаляются по истечении определенного времени.
import java.util.concurrent.DelayQueue;

public class CacheExample {
private static class CacheItem implements Delayed {
private final String value;
private final long expiryTime;

public CacheItem(String value, long delay) {
this.value = value;
this.expiryTime = System.currentTimeMillis() + delay;
}

@Override
public long getDelay(TimeUnit unit) {
long delay = expiryTime - System.currentTimeMillis();
return unit.convert(delay, TimeUnit.MILLISECONDS);
}

@Override
public int compareTo(Delayed o) {
if (this.expiryTime < ((CacheItem) o).expiryTime) {
return -1;
}
if (this.expiryTime > ((CacheItem) o).expiryTime) {
return 1;
}
return 0;
}

@Override
public String toString() {
return value;
}
}

public static void main(String[] args) throws InterruptedException {
DelayQueue<CacheItem> cache = new DelayQueue<>();

// Добавление элементов в кэш
cache.put(new CacheItem("Item1", 2000)); // 2 секунды жизни
cache.put(new CacheItem("Item2", 4000)); // 4 секунды жизни

// Извлечение элементов по истечении срока жизни
while (!cache.isEmpty()) {
CacheItem item = cache.take();
System.out.println("Expired item: " + item);
}
}
}

#Java #Training #Medium #DelayQueue
Утренний анекдот🫣

https://t.me/Java_for_beginner_dev

#Mems
Всем привет!
(Установка активированной IntelliJ IDEA ideaIU-2024.2.0.1 Ulimate)

В связи с тем, что при отсоединении чата от канала, потерлись комментарии всех ранее опубликованных постов, вновь запилю гайд по установке взломанной версии IntelliJ IDEA Ulimate, да и всех остальных продуктов JetBrains заодно. (которые не хотят взаимодействовать с программистами из России 😡).

Итак:
1. Качаем с официального сайта
https://www.jetbrains.com/ версию того продукта который вам нужен. Я использовал новейшую IntelliJ IDEA ideaIU-2024.2.0.1 (на момент написания поста), поэтому приложу ее к посту. (Кому-то может понадобиться VPN).
2. Качаем, приложенный в комментариях архив
jetbra.zip
3. Распаковываем
jetbra.zip в нужное вам место на Вашем компуктере, переходим в папку scripts и выполняем нужный скрипт и ждем окончания его работы (окошечко с кнопкой done):
- install-current-user.vbs — для текущего пользователя Windows
- install-all-users.vbs — для всех пользователей Windows
-
install.sh — для MacOS и Linux
4. В папке расположения продукта (обычно такой путь - C:\Program Files\JetBrains\IntelliJ IDEA 2024.2.0.1\bin) находим файл idea64.exe.vmoptions и в нем проверяем наличие строки:
-javaagent:C:\jetbra\ja-netfilter.jar=jetbrains. Если ее нет, дописываем ручками! (Внимательно проверяйте путь, по которому у Вас находится распакованный архив jetbra.zip).
5. По выбору: из приложенного в комментариях файла получаем ключ (строго для IntelliJ IDEA), или переходим на сайт
https://3.jetbra.in, где в любом из активных хостов, выбираем необходимый для активации продукт и получаем в буфер ключ.
6. Запускаем продукт и переходим в вкладку Activation Code и вставляем скопированный ранее ключ.

Все! Ваш продукт активирован до 2026 года, что как я считаю более чем достаточно)))


Приятного использования и всем пожалуйста😉

#Installing_IDEA
Java for Beginner pinned «Всем привет! (Установка активированной IntelliJ IDEA ideaIU-2024.2.0.1 Ulimate) В связи с тем, что при отсоединении чата от канала, потерлись комментарии всех ранее опубликованных постов, вновь запилю гайд по установке взломанной версии IntelliJ IDEA Ulimate…»
SynchronousQueue

SynchronousQueue — это специализированная реализация интерфейса BlockingQueue, который позволяет передавать элементы между потоками напрямую. В отличие от других очередей, у SynchronousQueue нет внутреннего буфера или емкости. Каждый put-запрос должен соответствовать take-запросу, чтобы операция завершилась. Это делает SynchronousQueue полезной в ситуациях, когда требуется прямая передача данных между потоками.

Внутреннее устройство SynchronousQueue

SynchronousQueue можно представить как точку передачи данных, где один поток передает элемент, и другой поток немедленно его забирает. Если нет готового потребителя для передачи, операция будет заблокирована до тех пор, пока потребитель не появится.

Блокировка и синхронизация:

SynchronousQueue использует два режима передачи данных: "fair" (справедливый) и "non-fair" (несправедливый). В справедливом режиме потоки обслуживаются в порядке поступления (FIFO), тогда как в несправедливом режиме порядок не гарантируется.
Реализация на основе блокировки и синхронизации происходит с помощью ReentrantLock и Condition.


Режимы работы:

Fair Mode: Гарантирует обслуживание потоков в порядке поступления. Это достигается за счет использования очередей ожидания для операций put и take.
Non-Fair Mode: Потоки обслуживаются без строгого порядка, что может приводить к уменьшению времени ожидания для некоторых потоков, но не гарантирует равномерности.


Основные структуры данных:

В справедливом режиме используются две очереди ожидания для хранения ожидающих put и take операций.
В несправедливом режиме используется стек для выполнения операций.


Пример кода внутреннего устройства
import java.util.concurrent.SynchronousQueue;
import java.util.concurrent.TimeUnit;

public class SynchronousQueueExample {
public static void main(String[] args) {
SynchronousQueue<Integer> queue = new SynchronousQueue<>(true); // fair mode

Thread producer = new Thread(() -> {
try {
System.out.println("Putting element: 1");
queue.put(1);
System.out.println("Element 1 put");
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
});

Thread consumer = new Thread(() -> {
try {
System.out.println("Taking element...");
Integer element = queue.take();
System.out.println("Element taken: " + element);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
});

producer.start();
consumer.start();
}
}


Особенности SynchronousQueue

Отсутствие буферизации: У SynchronousQueue нет внутреннего буфера, что означает, что каждое добавление элемента должно совпадать с извлечением.
Потокобезопасность: Очередь разработана с учетом многопоточности и обеспечивает высокую степень безопасности для работы в конкурентной среде.
Высокая скорость передачи данных: За счет отсутствия буферизации и немедленной передачи элементов достигается высокая скорость обмена данными между потоками.
Гибкость: Возможность выбора между справедливым и несправедливым режимами работы позволяет оптимизировать производительность в зависимости от конкретных требований.


Преимущества SynchronousQueue

Эффективность в межпоточном взаимодействии: SynchronousQueue идеально подходит для ситуаций, где требуется немедленная передача данных между потоками, например, для реализации паттерна "производитель-потребитель".
Простота реализации: Очередь предоставляет простой API для обмена данными между потоками, что упрощает разработку многопоточных приложений.
Высокая производительность: За счет минимального количества операций по синхронизации достигается высокая производительность при обмене данными.


Ссылки на полезные статьи (спасибо авторам за проделанную работу) :
https://www.baeldung.com/java-synchronous-queue

#Java #Training #Medium #SynchronousQueue