Java for Beginner
760 subscribers
744 photos
212 videos
12 files
1.24K links
Канал от новичков для новичков!
Изучайте Java вместе с нами!
Здесь мы обмениваемся опытом и постоянно изучаем что-то новое!

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

Наш канал на RUTube - https://rutube.ru/channel/37896292/
Download Telegram
История IT-технологий сегодня — 13 ноября


ℹ️ Кто родился в этот день

Талгат Касимович Сиразетдинов (родился в деревне Сауш Тюлячинского района Татарстана 13 ноября 1927 года) — советский/российский учёный в теории управления и математическом моделировании; его работы по управлению и моделированию применимы в управлении автоматизированными системами и смежных ИТ-решениях.

Скотт Г. Макнили (родился 13 ноября 1954 года) — сооснователь Sun Microsystems; как CEO и соавтор стратегии «The Network is the Computer» сыграл ключевую роль в развитии рабочих станций, серверов и сетевой инфраструктуры 1980–2000-х.

Пол Грэм (англ. Paul Graham, родился 13 ноября 1964, Уэймут) — программист, автор диалекта Lisp (Arc), сооснователь Viaweb (позже Yahoo! Store) и акселератора Y Combinator; влиятельный популяризатор идей стартапов и функционального программирования.


🌐 Знаковые события

1971 — состоялся вывод на орбиту вокруг Марса первого межпланетного космического аппарата; первый искусственный спутник Марса («Маринер-9», США).


#Biography #Birth_Date #Events #13Ноября
Please open Telegram to view this post
VIEW IN TELEGRAM
👍4
Архитектура и принципы работы GraphQL
Что происходит между клиентом, схемой и источниками данных


1. Главная идея архитектуры

GraphQL-сервер — это прослойка, которая принимает декларативные запросы, проверяет их на соответствие схеме, исполняет нужные резолверы и возвращает результат.

Он не хранит данные сам по себе.
GraphQL — это не база данных, а универсальный интерфейс к любому источнику данных: SQL, NoSQL, микросервисы, REST, файлы, внешние API.
Его задача — связать клиентскую структуру запроса со структурой данных в бэкенде.


Схематично:
Клиент → GraphQL сервер → Источники данных (DB, REST, gRPC, API)


2. Schema — сердце GraphQL

Schema (схема) — это основной контракт между клиентом и сервером.

Она описывает:
какие данные доступны;
какие поля у этих данных есть;
какие операции можно выполнять.


Схема написана на SDL (Schema Definition Language), декларативном языке, напоминающем описание классов.

Пример:
type User {
id: ID!
name: String!
posts: [Post!]!
}

type Post {
id: ID!
title: String!
content: String
}

type Query {
user(id: ID!): User
allUsers: [User!]!
}


Ключевые понятия:
type — определяет структуру данных (аналог Java-класса или DTO).
! — обязательное поле (non-null).
[User!]! — список пользователей, где ни один элемент не равен null.
Query — специальный тип, описывающий операции чтения.


Схема — это не код, а декларация.
На её основе GraphQL сервер автоматически понимает, как валидировать запросы, какие поля допустимы и какие возвращать ошибки.



3. Query, Mutation, Subscription — три типа операций

GraphQL разделяет все действия на три категории.

1. Query — запрос данных (аналог GET в REST)
Используется для чтения.

Клиент описывает, какие сущности и поля нужны:
query {
user(id: 42) {
name
posts {
title
}
}
}


Сервер вернёт данные в том же формате:
{
"data": {
"user": {
"name": "Den",
"posts": [
{ "title": "gRPC и Java" },
{ "title": "GraphQL под капотом" }
]
}
}
}


2. Mutation — изменение данных (аналог POST/PUT/DELETE)
Mutation обозначает действия, которые модифицируют состояние системы.
Они могут создавать, обновлять или удалять записи.

mutation {
createUser(input: { name: "Alex" }) {
id
name
}
}


GraphQL возвращает результат — обновлённые данные или подтверждение операции.

3. Subscription — подписка на события (реактивный поток)
Subscription создаёт постоянное соединение между клиентом и сервером (обычно через WebSocket).
Когда на сервере происходят изменения, клиент получает уведомления в реальном времени.

subscription {
userCreated {
id
name
}
}


Если на сервере создан новый пользователь, событие userCreated автоматически отправляется всем подписанным клиентам.


4. Как работает запрос GraphQL: путь от клиента до данных

Чтобы понять механику, посмотрим на полный цикл обработки запроса.

Шаг 1. Клиент отправляет запрос

Клиент (например, браузер) отправляет HTTP POST на /graphql с JSON-телом:
{
"query": "query { user(id: 42) { name posts { title } } }"
}

GraphQL-запрос — это декларативное описание структуры данных, не код и не SQL.


Шаг 2. Сервер парсит и валидирует запрос

GraphQL-сервер:
Парсит текст запроса.
Проверяет, что все поля и типы существуют в схеме.
Проверяет типы аргументов (id действительно ID!).
Отклоняет запрос, если нарушен контракт схемы.


Таким образом, сервер никогда не выполнит запрос, который не соответствует схеме.
Это гарантирует типобезопасность и предсказуемость работы.



#Java #middle #GraphQL
👍1
Шаг 3. Сервер вызывает резолверы

Резолвер (resolver) — это функция, которая знает, как получить данные для конкретного поля.


Например, в Java через библиотеку graphql-java:
GraphQLObjectType userType = newObject()
.name("User")
.field(field -> field
.name("id")
.type(Scalars.GraphQLID))
.field(field -> field
.name("name")
.type(Scalars.GraphQLString))
.field(field -> field
.name("posts")
.type(new GraphQLList(postType))
.dataFetcher(env -> postService.getByUser(env.getSource())))
.build();



Резолверы вызываются рекурсивно:

Сначала выполняется user(id: 42).
Затем для каждого пользователя — posts.
Затем для каждого поста — title.

GraphQL умеет оптимизировать выполнение: например, группировать одинаковые вызовы (DataLoader паттерн).


Шаг 4. Формирование ответа

После выполнения всех резолверов GraphQL собирает результат в структуру, повторяющую форму запроса, и отправляет JSON-ответ клиенту:
{
"data": {
"user": {
"name": "Den",
"posts": [{ "title": "gRPC и Java" }]
}
}
}


Если на каком-то шаге произошла ошибка, она не прерывает всё выполнение.

GraphQL вернёт частичные данные + список ошибок:
{
"data": { "user": null },
"errors": [{ "message": "User not found" }]
}



5. Почему GraphQL — не база данных

Эта путаница встречается часто.
GraphQL не хранит и не управляет данными.
Он не заменяет SQL или ORM.


GraphQL — это API-уровень, который:
принимает запросы,
интерпретирует их,
вызывает нужные источники данных,
собирает и возвращает результат в едином формате.


Под капотом это может быть:
JDBC-запросы в PostgreSQL,
вызовы REST API других микросервисов,
gRPC-вызовы,
кэш Redis,
файловая система или внешние API.

GraphQL — унифицированный интерфейс доступа к любым данным, независимо от их источника.



6. Интеграция GraphQL в бэкенд

GraphQL не заменяет ваш backend — он становится поверх него.

В Java это обычно выглядит так:
Spring Boot + graphql-spring-boot-starter — стандартный способ поднять GraphQL-сервер.
Схема (.graphqls) описывается декларативно.
Резолверы реализуются как обычные Spring-бины.


Пример:
@Component
public class UserResolver implements GraphQLQueryResolver {
private final UserService userService;
public UserResolver(UserService userService) { this.userService = userService; }

public User user(Long id) {
return userService.findById(id);
}

public List<User> allUsers() {
return userService.findAll();
}
}


GraphQL сам вызывает нужный метод в зависимости от запроса клиента.

Таким образом, GraphQL интегрируется поверх существующего слоя сервисов и репозиториев, не требуя переписывания бизнес-логики.


#Java #middle #GraphQL
👍1
Что выведет код?

import java.util.concurrent.ConcurrentHashMap;

public class Task131125 {
public static void main(String[] args) {
ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();
map.put("a", 1);
map.put("b", 2);
map.put("c", 3);

int result = map.reduce(2,
(k, v) -> k.length() + v,
Integer::sum
);

System.out.println(result);
}
}


#Tasks
Варианты ответа:
Anonymous Quiz
30%
6
20%
9
50%
12
0%
15
Вопрос с собеседований

Что такое CI/CD pipeline? 🤓


Ответ:

CI/CD
— это процесс автоматизации сборки, тестирования и доставки кода.

CI (Continuous Integration) проверяет код при каждом коммите, CD (Continuous Deployment/Delivery) — автоматически выкатывает его на сервер.

Примеры: Jenkins, GitHub Actions, GitLab CI.



#собеседование
Please open Telegram to view this post
VIEW IN TELEGRAM
👍3
История IT-технологий сегодня — 14 ноября


ℹ️ Кто родился в этот день

Александр Юльевич Дроздов (род. 14 ноября 1966) — российский разработчик/учёный, кандидат/д.т.н.; вёл работы по трансляторам, автоматическим распараллеливателям и созданию Универсальной Библиотеки Трансляции (УБТ), применимым к современным архитектурам (x86, ARM и др.).

Питер Нортон (англ. Peter Norton; род. 14 ноября 1943)американский предприниматель, программист и филантроп. В 1980-х годах создал популярную программу для персональных компьютеров, позволяющую восстанавливать стёртые данные. Эта утилита вместе с несколькими другими образовали пакет Norton Utilities. Вскоре в пакет был включён Norton Commander, удобный файловый менеджер для операционной системы DOS.


🌐 Знаковые события

1969 — с мыса Канаверал стартовал к Луне американский космический корабль «Аполлон-12», пилотируемый астронавтами Ч. Конрадом, А. Бином и Р. Гордоном.


#Biography #Birth_Date #Events #14Ноября
Please open Telegram to view this post
VIEW IN TELEGRAM
👍3
Раздел 6. Коллекции в Java

Глава 5. Map — отображения (словари)

Основные методы: Итерация по Map: entrySet, keySet, values

Итерация по Map — это не просто последовательный перебор элементов, а сложный процесс навигации по внутренней структуре данных, который должен учитывать специфику реализации, обеспечивать корректность при concurrent модификациях и предоставлять различные perspectives на данные. Три основных view — entrySet, keySet и values — представляют собой разные проекции одного и того же набора данных, каждая из которых оптимизирована для определенных сценариев использования.


Общая архитектура представлений (Views)

Представления в Map реализованы по принципу lazy initialization и являются "окнами" в основную структуру данных. Они не содержат собственных копий элементов, а предоставляют live view содержимого Map. Это создает как преимущества в виде экономии памяти и мгновенного отражения изменений, так и challenges в обеспечении consistency и производительности.


Модель отложенной инициализации

Все три представления обычно создаются по требованию при первом вызове соответствующих методов.

Механизм их работы строится на следующих принципах:
Легковесность: Представления не дублируют данные, а содержат ссылки на исходную Map
Синхронизация изменений: Модификации в Map немедленно отражаются в представлениях
Разделяемая состояние: Несколько представлений разделяют общее состояние с родительской Map
Делегирование операций: Все методы представлений делегируются к внутренней структуре Map



Метод entrySet()

Метод entrySet() возвращает представление пар "ключ-значение" в виде Set объектов Map.Entry. Это наиболее полное и мощное представление, предоставляющее доступ как к ключам, так и к значениям, а также возможность модификации значений через интерфейс Map.Entry.

Внутренняя механика работы

Структура данных представления:
Для каждой реализации Map создается специализированная реализация Set, которая инкапсулирует логику обхода внутренней структуры данных. Эта реализация содержит ссылку на родительскую Map и делегирует ей все операции.

Процесс итерации:
Итератор для entrySet должен на каждом шаге извлекать как ключ, так и значение, что требует coordinated доступа к внутренним структурам данных.

Процесс итерации варьируется в зависимости от реализации:
В HashMap: Итератор проходит по массиву бакетов, для каждого непустого бакета обходит цепочку коллизий (связный список или дерево), создавая объекты Map.Entry для каждого элемента
В TreeMap: Итератор выполняет обход красно-черного дерева в порядке inorder traversal, гарантируя сортированный порядок элементов
В LinkedHashMap: Итератор следует по двусвязному списку, поддерживающему порядок добавления или доступа


Механизм создания Map.Entry:
Объекты Map.Entry, возвращаемые итератором, обычно являются view objects, которые не хранят данные самостоятельно, а содержат ссылки на внутренние узлы структуры данных. Это позволяет эффективно обновлять значения через метод setValue().


Особенности производительности

Временная сложность: Полный обход через entrySet имеет сложность O(n), где n — количество элементов в Map. Однако константные множители значительно различаются между реализациями.
Потребление памяти: entrySet создает временные объекты Map.Entry во время итерации, что может создавать pressure на garbage collector при обходе больших коллекций.
Оптимизации: Современные JVM применяют escape analysis и stack allocation для минимизации overhead создания временных объектов.



#Java #для_новичков #beginner #Map #entrySet #keySet #values
Метод keySet()

Метод keySet() возвращает представление ключей Map в виде Set. Это представление фокусируется исключительно на ключах, предоставляя упрощенный view данных, который полезен для операций проверки принадлежности, массового удаления и других операций, ориентированных на ключи.

Внутренняя механика работы

Архитектура представления:

keySet реализуется как специализированный Set, который делегирует все операции родительской Map. Критически важным аспектом является то, что операции удаления через keySet непосредственно влияют на исходную Map.

Процесс итерации:
Итератор keySet извлекает только ключи, пропуская значения.

Это может быть более эффективно в сценариях, где значения не нужны:
В HashMap: Итератор обходит ту же структуру бакетов, что и entrySet, но возвращает только ключевую компоненту
В TreeMap: Обход дерева выполняется аналогично entrySet, но с извлечением только ключей
В LinkedHashMap: Следование по связному списку с возвратом ключей


Операция удаления через итератор:
При вызове remove() итератора keySet происходит удаление соответствующей пары "ключ-значение" из Map. Этот процесс требует локализации и удаления всего узла, а не только ключа.

Особенности производительности
Эффективность обхода: keySet может быть более эффективен, чем entrySet, когда требуются только ключи, так как избегает создания объектов Map.Entry и извлечения значений.
Операции массового удаления: Методы removeAll() и retainAll() в keySet оптимизированы для работы с ключами и могут быть более эффективны, чем эквивалентные операции через entrySet.
Потребление памяти: keySet обычно создает меньше временных объектов во время итерации по сравнению с entrySet.



Метод values()

Метод values() возвращает представление значений Map в виде Collection. Это представление фокусируется исключительно на значениях, предоставляя view, которое полезно для операций обработки значений, статистического анализа и преобразований.

Внутренняя механика работы

Архитектура представления:
values возвращает Collection, а не Set, поскольку значения могут содержать дубликаты. Эта коллекция поддерживает только операции итерации и удаления, но не добавления, так как значения не могут существовать без ключей.

Процесс итерации:

Итератор values извлекает только значения, что может быть наиболее эффективно в сценариях, где требуются исключительно значения:
В HashMap: Итератор проходит по бакетам и цепочкам, извлекая только value компоненту узлов
В TreeMap: Обход дерева с возвратом значений в порядке сортировки ключей
В LinkedHashMap: Следование по списку с извлечением значений


Особенности модификации:
Коллекция values поддерживает удаление элементов через итератор и методы коллекции. Однако операция удаления по значению требует поиска соответствующего ключа, что может быть затратным.

Особенности производительности
Эффективность для value-oriented операций: values является наиболее эффективным представлением для операций, ориентированных исключительно на значения, таких как статистические вычисления, агрегации и преобразования.
Сложность операций удаления: Удаление по значению требует поиска ключа, ассоциированного с данным значением, что может иметь сложность O(n) в худшем случае.
Отсутствие гарантий уникальности: Поскольку значения могут дублироваться, итерация через values может возвращать повторяющиеся элементы.



Сравнительный анализ представлений

Производительность итерации

Временная сложность:
Все три представления имеют одинаковую асимптотическую сложность O(n) для полного обхода, но различаются константными множителями:
entrySet: Наиболее универсален, но создает наибольшее количество временных объектов
keySet: Более эффективен при работе только с ключами, уменьшает overhead
values: Наиболее эффективен при работе только со значениями


Потребление памяти:
entrySet: Создает временные объекты Map.Entry
keySet: Минимальное потребление памяти при итерации
values: Сравнимо с keySet по потреблению памяти


#Java #для_новичков #beginner #Map #entrySet #keySet #values
Семантика модификаций

Влияние на исходную Map:
Все три представления предоставляют live view, и модификации через них непосредственно влияют на исходную Map:
Удаление через любой итератор удаляет соответствующую пару из Map
Изменение значений через entrySet изменяет значения в Map
Очистка представления очищает исходную Map


Ограничения модификаций:

entrySet: Поддерживает модификацию значений через Map.Entry.setValue()
keySet: Поддерживает только удаление элементов
values: Поддерживает только удаление элементов



Специфика реализации в различных Map

HashMap и связанные реализации

Структура итератора:
Итераторы в HashMap должны обрабатывать сложную структуру данных, включающую массив бакетов, связные списки и деревья.

Процесс итерации включает:
Поиск следующего непустого бакета
Навигацию по цепочке коллизий (список или дерево)
Обработку структурных изменений во время итерации


Механизм fail-fast:
Итераторы HashMap используют счетчик modCount для обнаружения структурных изменений во время итерации. При обнаружении неавторизованной модификации выбрасывается ConcurrentModificationException.

Оптимизации Java 8+:
В современных версиях HashMap итераторы эффективно работают с hybrid структурами, автоматически адаптируясь к спискам и деревьям.


TreeMap

Упорядоченная итерация:


TreeMap обеспечивает обход элементов в sorted порядке, что достигается через:
Inorder traversal красно-черного дерева
Эффективные алгоритмы навигации между узлами
Поддержку descending итераторов

Балансировка и итерация:
Процесс итерации должен корректно работать в условиях ongoing балансировки дерева, обеспечивая consistency обхода.


LinkedHashMap

Итерация с сохранением порядка:
LinkedHashMap гарантирует итерацию в порядке добавления или доступа, что реализуется через:
Следование по двусвязному списку
Поддержку access-order при итерации
Эффективное обновление порядка при операциях доступа



ConcurrentHashMap


Потокобезопасная итерация:
ConcurrentHashMap предоставляет weakly consistent итераторы, которые:
Не выбрасывают ConcurrentModificationException
Могут отражать только часть изменений, произошедших после создания итератора
Обеспечивают высокую производительность в многопоточной среде


Сегментированный обход:
Итерация в ConcurrentHashMap может выполняться по сегментам, что позволяет параллельную обработку в некоторых сценариях.


Потокобезопасность и concurrent модификации

Модель fail-fast

Большинство несинхронизированных реализаций Map используют fail-fast итераторы, которые:
Выбрасывают ConcurrentModificationException при обнаружении структурных изменений
Основаны на сравнении счетчика modCount
Обеспечивают раннее обнаружение ошибок синхронизации


Weakly consistent итераторы
ConcurrentHashMap и другие concurrent реализации используют weakly consistent итераторы, которые:
Не гарантируют отражение всех последних изменений
Не выбрасывают исключения при concurrent модификациях
Обеспечивают баланс между performance и consistency


#Java #для_новичков #beginner #Map #entrySet #keySet #values
Что выведет код?

import java.util.HashMap;
import java.util.Map;

public class Task131125 {
public static void main(String[] args) {
Map<String, Integer> map = new HashMap<>();
map.put("a", 1);
map.put("b", 2);

map.keySet().remove("a");
map.values().remove(2);
map.entrySet().removeIf(entry -> entry.getKey().equals("c"));

System.out.println(map.size());
System.out.println(map.get("a"));
System.out.println(map.get("b"));
}
}


#Tasks
Варианты ответа:
Anonymous Quiz
33%
0 null null
33%
1 null 2
33%
0 null 2
0%
1 null null