Коллекции в Java
Глава 3. Set — множества
Интерфейс Set. Особенности множеств
Интерфейс Set<E> — это часть Java Collections Framework (JCF), который представляет коллекцию уникальных элементов без дубликатов. Set является подинтерфейсом Collection, но с ключевым отличием: он не позволяет хранить одинаковые элементы. Если вы пытаетесь добавить дубликат, операция игнорируется.
Основные характеристики Set:
Уникальность элементов: Да, дубликаты не хранятся.
Упорядоченность: В общем случае нет (зависит от реализации).
Сортировка: Нет по умолчанию, но возможна в подтипах.
Время доступа (Big O): Для основных операций (add, remove, contains) — O(1) в HashSet, O(log n) в TreeSet.
Set моделирует математическое множество: элементы уникальны, нет индексации, фокус на наличии/отсутствии.
Методы Set (наследуются от Collection, но с нюансами):
boolean add(E e): Добавляет элемент, если его нет (возвращает true, если добавлен).
boolean remove(Object o): Удаляет элемент, если он есть.
boolean contains(Object o): Проверяет наличие.
int size(): Размер.
boolean isEmpty(): Пустота.
Iterator<E> iterator(): Для перебора (нет порядка по умолчанию).
void clear(): Очистка.
Нюанс: Set не имеет get(int index) — нет индексации, как в List. Перебор через Iterator или for-each.
Особенности множеств в Java
Множества в Java имеют несколько важных особенностей, которые определяют их использование:
Уникальность элементов:
Set автоматически предотвращает дубликаты на основе методов equals() и hashCode() (для HashSet) или compareTo() (для TreeSet).
Если добавить существующий элемент, add() возвращает false, коллекция не меняется.
Нюанс: Для custom классов обязательно переопределите equals() и hashCode() (используйте Objects.equals() и Objects.hash()). Без этого уникальность по ссылке, не по значению.
Отсутствие гарантированного порядка:
В HashSet порядок непредсказуем (зависит от хэша).
В LinkedHashSet — порядок вставки.
В TreeSet — сортированный порядок.
Нюанс: Не полагайтесь на порядок в HashSet — он может измениться при ресайзе.
Null элементы:
HashSet и LinkedHashSet позволяют один null.
TreeSet — нет (NullPointerException, так как сравнивает).
Нюанс: Null в Set — редко рекомендуется, но возможно.
Итерация и модификация:
Перебор через for-each или Iterator.
Нюанс: Во время итерации нельзя модифицировать Set (ConcurrentModificationException). Используйте Iterator.remove() для удаления.
Производительность:
HashSet: O(1) для add/remove/contains (средний случай).
TreeSet: O(log n), но с автосортировкой.
Нюанс: HashSet требует хорошего hashCode() — плохой приводит к O(n) worst-case.
Синхронизация:
Стандартные реализации не thread-safe. Для многопоточности: Collections.synchronizedSet(Set set) или ConcurrentHashSet (из Guava).
#Java #для_новичков #beginner #Collections #Set
Глава 3. Set — множества
Интерфейс Set. Особенности множеств
Интерфейс Set<E> — это часть Java Collections Framework (JCF), который представляет коллекцию уникальных элементов без дубликатов. Set является подинтерфейсом Collection, но с ключевым отличием: он не позволяет хранить одинаковые элементы. Если вы пытаетесь добавить дубликат, операция игнорируется.
Основные характеристики Set:
Уникальность элементов: Да, дубликаты не хранятся.
Упорядоченность: В общем случае нет (зависит от реализации).
Сортировка: Нет по умолчанию, но возможна в подтипах.
Время доступа (Big O): Для основных операций (add, remove, contains) — O(1) в HashSet, O(log n) в TreeSet.
Set моделирует математическое множество: элементы уникальны, нет индексации, фокус на наличии/отсутствии.
Методы Set (наследуются от Collection, но с нюансами):
boolean add(E e): Добавляет элемент, если его нет (возвращает true, если добавлен).
boolean remove(Object o): Удаляет элемент, если он есть.
boolean contains(Object o): Проверяет наличие.
int size(): Размер.
boolean isEmpty(): Пустота.
Iterator<E> iterator(): Для перебора (нет порядка по умолчанию).
void clear(): Очистка.
Нюанс: Set не имеет get(int index) — нет индексации, как в List. Перебор через Iterator или for-each.
Особенности множеств в Java
Множества в Java имеют несколько важных особенностей, которые определяют их использование:
Уникальность элементов:
Set автоматически предотвращает дубликаты на основе методов equals() и hashCode() (для HashSet) или compareTo() (для TreeSet).
Если добавить существующий элемент, add() возвращает false, коллекция не меняется.
Нюанс: Для custom классов обязательно переопределите equals() и hashCode() (используйте Objects.equals() и Objects.hash()). Без этого уникальность по ссылке, не по значению.
Отсутствие гарантированного порядка:
В HashSet порядок непредсказуем (зависит от хэша).
В LinkedHashSet — порядок вставки.
В TreeSet — сортированный порядок.
Нюанс: Не полагайтесь на порядок в HashSet — он может измениться при ресайзе.
Null элементы:
HashSet и LinkedHashSet позволяют один null.
TreeSet — нет (NullPointerException, так как сравнивает).
Нюанс: Null в Set — редко рекомендуется, но возможно.
Итерация и модификация:
Перебор через for-each или Iterator.
Нюанс: Во время итерации нельзя модифицировать Set (ConcurrentModificationException). Используйте Iterator.remove() для удаления.
Производительность:
HashSet: O(1) для add/remove/contains (средний случай).
TreeSet: O(log n), но с автосортировкой.
Нюанс: HashSet требует хорошего hashCode() — плохой приводит к O(n) worst-case.
Синхронизация:
Стандартные реализации не thread-safe. Для многопоточности: Collections.synchronizedSet(Set set) или ConcurrentHashSet (из Guava).
#Java #для_новичков #beginner #Collections #Set
👍6
Примеры использования Set
HashSet: Для быстрых операций без порядка.
LinkedHashSet: С сохранением порядка вставки.
TreeSet: С автосортировкой.
Нюанс: Для custom классов в TreeSet реализуйте Comparable или используйте Comparator при создании.
Как создать и использовать Set в IntelliJ IDEA
Импорт: В коде напишите Set — IDE предложит import java.util.Set; и реализацию (HashSet и т.д.).
Generics: Используйте Set для типобезопасности.
Автодополнение: При add() IDE подскажет параметры.
Отладка: В debug смотрите содержимое Set — IDE покажет элементы.
Конвертация: Из List в Set: new HashSet<>(list) — для удаления дубликатов.
Полезные советы для новичков
Выбор реализации: HashSet для скорости, LinkedHashSet для порядка, TreeSet для сортировки.
equals() и hashCode(): Всегда переопределяйте в custom классах для Set/Map (используйте @Override и Objects.hash()).
Удаление дубликатов: Set — быстрый способ: new HashSet<>(list).
Null: Избегайте в TreeSet; в HashSet — осторожно.
Итерация: For-each безопасен для чтения, но не модифицируйте во время перебора.
#Java #для_новичков #beginner #Collections #Set
HashSet: Для быстрых операций без порядка.
import java.util.HashSet;
import java.util.Set;
public class Main {
public static void main(String[] args) {
Set<String> fruits = new HashSet<>();
fruits.add("Яблоко");
fruits.add("Банан");
fruits.add("Яблоко"); // Игнорируется
System.out.println(fruits.size()); // 2
System.out.println(fruits.contains("Банан")); // true
for (String fruit : fruits) {
System.out.println(fruit); // Порядок непредсказуем
}
}
}
Вывод: Размер 2, содержит "Банан", элементы в случайном порядке.
LinkedHashSet: С сохранением порядка вставки.
import java.util.LinkedHashSet;
import java.util.Set;
public class Main {
public static void main(String[] args) {
Set<String> fruits = new LinkedHashSet<>();
fruits.add("Яблоко");
fruits.add("Банан");
fruits.add("Апельсин");
for (String fruit : fruits) {
System.out.println(fruit); // Яблоко, Банан, Апельсин — порядок вставки
}
}
}
TreeSet: С автосортировкой.
import java.util.TreeSet;
import java.util.Set;
public class Main {
public static void main(String[] args) {
Set<Integer> numbers = new TreeSet<>();
numbers.add(5);
numbers.add(1);
numbers.add(3);
for (Integer num : numbers) {
System.out.println(num); // 1, 3, 5 — отсортировано
}
}
}
Нюанс: Для custom классов в TreeSet реализуйте Comparable или используйте Comparator при создании.
Как создать и использовать Set в IntelliJ IDEA
Импорт: В коде напишите Set — IDE предложит import java.util.Set; и реализацию (HashSet и т.д.).
Generics: Используйте Set для типобезопасности.
Автодополнение: При add() IDE подскажет параметры.
Отладка: В debug смотрите содержимое Set — IDE покажет элементы.
Конвертация: Из List в Set: new HashSet<>(list) — для удаления дубликатов.
Полезные советы для новичков
Выбор реализации: HashSet для скорости, LinkedHashSet для порядка, TreeSet для сортировки.
equals() и hashCode(): Всегда переопределяйте в custom классах для Set/Map (используйте @Override и Objects.hash()).
Удаление дубликатов: Set — быстрый способ: new HashSet<>(list).
Null: Избегайте в TreeSet; в HashSet — осторожно.
Итерация: For-each безопасен для чтения, но не модифицируйте во время перебора.
#Java #для_новичков #beginner #Collections #Set
👍4
Коллекции в Java
Глава 3. Set — множества
Методы add, remove, contains
Методы add, remove и contains — это основные операции для добавления, удаления и проверки элементов в Set. Они наследуются от интерфейса Collection, но в Set имеют специфику из-за уникальности элементов.
add(E e): Добавляет элемент в множество, если его еще нет. Возвращает boolean: true, если добавлен (элемент был уникален), false — если уже существовал (дубликат игнорируется).
remove(Object o): Удаляет элемент, если он существует. Возвращает boolean: true, если удален (элемент был), false — если не найден.
contains(Object o): Проверяет, содержит ли множество элемент. Возвращает boolean: true, если есть, false — если нет.
Эти методы работают за O(1) в HashSet/LinkedHashSet (средний случай) и O(log n) в TreeSet.
Общие нюансы:
Все методы используют equals() для сравнения элементов (и hashCode() в Hash-based реализациях).
Null: Разрешен в HashSet/LinkedHashSet (один), но в TreeSet вызывает NullPointerException при сравнении.
Изменение множества: Методы модифицируют Set in-place (на месте).
Thread-safety: Не гарантирована — используйте synchronized версии для многопоточности.
Generics: Set add(Integer) — ошибка компиляции (типобезопасность).
Метод add(E e): Добавление элементов
add() — основной способ наполнения Set. Если элемент уже есть (по equals()), он не добавляется, и возвращается false.
Поведение в реализациях:
HashSet: Добавляет в хэш-таблицу. Если хэш-коллизия, проверяет equals() в цепочке.
LinkedHashSet: Аналогично HashSet, но обновляет ссылки в списке для порядка вставки (только если добавлен).
TreeSet: Добавляет в дерево, сравнивая через compareTo() или Comparator. Если равен 0 — не добавляет.
Возвращаемое значение: true — добавлен (новый), false — уже был.
Исключения: NullPointerException в TreeSet для null; ClassCastException в TreeSet, если элемент не Comparable.
Нюансы:
Если Set полный (редко, так как resizable), может быть OutOfMemoryError.
Для custom объектов: Без правильного equals()/hashCode() может добавить "дубликаты" (по значению, но не по ссылке).
Модификация объекта после добавления: Не изменяйте поля, влияющие на equals/hashCode (например, в HashSet объект может "потеряться").
Пример кода для add():
#Java #для_новичков #beginner #Collections #Set #add #remove #contains
Глава 3. Set — множества
Методы add, remove, contains
Методы add, remove и contains — это основные операции для добавления, удаления и проверки элементов в Set. Они наследуются от интерфейса Collection, но в Set имеют специфику из-за уникальности элементов.
add(E e): Добавляет элемент в множество, если его еще нет. Возвращает boolean: true, если добавлен (элемент был уникален), false — если уже существовал (дубликат игнорируется).
remove(Object o): Удаляет элемент, если он существует. Возвращает boolean: true, если удален (элемент был), false — если не найден.
contains(Object o): Проверяет, содержит ли множество элемент. Возвращает boolean: true, если есть, false — если нет.
Эти методы работают за O(1) в HashSet/LinkedHashSet (средний случай) и O(log n) в TreeSet.
Общие нюансы:
Все методы используют equals() для сравнения элементов (и hashCode() в Hash-based реализациях).
Null: Разрешен в HashSet/LinkedHashSet (один), но в TreeSet вызывает NullPointerException при сравнении.
Изменение множества: Методы модифицируют Set in-place (на месте).
Thread-safety: Не гарантирована — используйте synchronized версии для многопоточности.
Generics: Set add(Integer) — ошибка компиляции (типобезопасность).
Метод add(E e): Добавление элементов
add() — основной способ наполнения Set. Если элемент уже есть (по equals()), он не добавляется, и возвращается false.
Поведение в реализациях:
HashSet: Добавляет в хэш-таблицу. Если хэш-коллизия, проверяет equals() в цепочке.
LinkedHashSet: Аналогично HashSet, но обновляет ссылки в списке для порядка вставки (только если добавлен).
TreeSet: Добавляет в дерево, сравнивая через compareTo() или Comparator. Если равен 0 — не добавляет.
Возвращаемое значение: true — добавлен (новый), false — уже был.
Исключения: NullPointerException в TreeSet для null; ClassCastException в TreeSet, если элемент не Comparable.
Нюансы:
Если Set полный (редко, так как resizable), может быть OutOfMemoryError.
Для custom объектов: Без правильного equals()/hashCode() может добавить "дубликаты" (по значению, но не по ссылке).
Модификация объекта после добавления: Не изменяйте поля, влияющие на equals/hashCode (например, в HashSet объект может "потеряться").
Пример кода для add():
import java.util.HashSet;
import java.util.LinkedHashSet;
import java.util.TreeSet;
import java.util.Set;
public class Main {
public static void main(String[] args) {
// HashSet
Set<String> hashSet = new HashSet<>();
boolean added1 = hashSet.add("Яблоко"); // true
boolean added2 = hashSet.add("Яблоко"); // false (дубликат)
System.out.println(hashSet); // [Яблоко]
// LinkedHashSet
Set<String> linkedSet = new LinkedHashSet<>();
linkedSet.add("Яблоко");
linkedSet.add("Банан");
linkedSet.add("Яблоко"); // false
System.out.println(linkedSet); // [Яблоко, Банан] — порядок вставки
// TreeSet
Set<Integer> treeSet = new TreeSet<>();
treeSet.add(5);
treeSet.add(1);
treeSet.add(5); // false
System.out.println(treeSet); // [1, 5] — отсортировано
}
}
Вывод показывает уникальность и особенности каждой реализации.
#Java #для_новичков #beginner #Collections #Set #add #remove #contains
👍5
Метод remove(Object o): Удаление элементов
remove() удаляет элемент, если он найден по equals(), и возвращает true/false.
Поведение в реализациях:
HashSet: Находит по хэшу, затем equals() — O(1).
LinkedHashSet: Аналогично, плюс обновляет ссылки списка.
TreeSet: Ищет по compareTo() — O(log n), удаляет узел дерева.
Возвращаемое значение: true — удален, false — не найден.
Исключения: NullPointerException в TreeSet для null; ClassCastException, если тип не совместим.
Нюансы:
Аргумент Object: Можно удалять по объекту любого типа, но сравнивает equals().
После remove: Размер уменьшается, итераторы обновляются.
Для custom: Зависит от equals().
Пример кода для remove():
Метод contains(Object o): Проверка наличия
contains() проверяет, есть ли элемент в Set по equals().
Поведение в реализациях:
HashSet: O(1) — хэш + equals().
LinkedHashSet: O(1), но с overhead списка.
TreeSet: O(log n) — поиск в дереве.
Возвращаемое значение: true — есть, false — нет.
Исключения: Аналогично remove: NPE в TreeSet для null.
Нюансы:
Быстрее, чем в List (O(n)), идеально для проверок уникальности.
Для больших Set: HashSet fastest.
Пример кода для contains():
Полезные советы для новичков
add для уникальности: Используйте возвращаемое значение для логики (if (!set.add(e)) { "Дубликат!"; }).
remove/contains для null: Тестируйте — в HashSet работает, в TreeSet — нет.
Custom объекты: Переопределяйте equals/hashCode (IDE: Generate → equals() and hashCode()).
Эффективность: Для частых contains — HashSet; для сортировки — TreeSet.
Комбинируйте: Set для фильтра, затем List для порядка.
Ошибки: ClassCastException в TreeSet без Comparable; ConcurrentModification при модификации в цикле (используйте Iterator).
#Java #для_новичков #beginner #Collections #Set #add #remove #contains
remove() удаляет элемент, если он найден по equals(), и возвращает true/false.
Поведение в реализациях:
HashSet: Находит по хэшу, затем equals() — O(1).
LinkedHashSet: Аналогично, плюс обновляет ссылки списка.
TreeSet: Ищет по compareTo() — O(log n), удаляет узел дерева.
Возвращаемое значение: true — удален, false — не найден.
Исключения: NullPointerException в TreeSet для null; ClassCastException, если тип не совместим.
Нюансы:
Аргумент Object: Можно удалять по объекту любого типа, но сравнивает equals().
После remove: Размер уменьшается, итераторы обновляются.
Для custom: Зависит от equals().
Пример кода для remove():
import java.util.HashSet;
import java.util.Set;
public class Main {
public static void main(String[] args) {
Set<String> fruits = new HashSet<>();
fruits.add("Яблоко");
fruits.add("Банан");
boolean removed1 = fruits.remove("Яблоко"); // true
boolean removed2 = fruits.remove("Апельсин"); // false (не найден)
System.out.println(fruits); // [Банан]
}
}
Аналогично для других реализаций: В TreeSet remove сохраняет сортировку.
Метод contains(Object o): Проверка наличия
contains() проверяет, есть ли элемент в Set по equals().
Поведение в реализациях:
HashSet: O(1) — хэш + equals().
LinkedHashSet: O(1), но с overhead списка.
TreeSet: O(log n) — поиск в дереве.
Возвращаемое значение: true — есть, false — нет.
Исключения: Аналогично remove: NPE в TreeSet для null.
Нюансы:
Быстрее, чем в List (O(n)), идеально для проверок уникальности.
Для больших Set: HashSet fastest.
Пример кода для contains():
import java.util.HashSet;
import java.util.Set;
public class Main {
public static void main(String[] args) {
Set<String> fruits = new HashSet<>();
fruits.add("Яблоко");
System.out.println(fruits.contains("Яблоко")); // true
System.out.println(fruits.contains("Банан")); // false
}
}
Полезные советы для новичков
add для уникальности: Используйте возвращаемое значение для логики (if (!set.add(e)) { "Дубликат!"; }).
remove/contains для null: Тестируйте — в HashSet работает, в TreeSet — нет.
Custom объекты: Переопределяйте equals/hashCode (IDE: Generate → equals() and hashCode()).
Эффективность: Для частых contains — HashSet; для сортировки — TreeSet.
Комбинируйте: Set для фильтра, затем List для порядка.
Ошибки: ClassCastException в TreeSet без Comparable; ConcurrentModification при модификации в цикле (используйте Iterator).
#Java #для_новичков #beginner #Collections #Set #add #remove #contains
👍5
Раздел 6. Коллекции в Java
Глава 3. Set — множества
Практика: В «Библиотеке» создать коллекцию Set для хранения уникальных имён авторов.
Добавлять автора при добавлении книги, проверять уникальность.
Откройте проект «Библиотека»: Запустите IntelliJ IDEA и откройте существующий проект LibraryProject (или как вы его назвали). Убедитесь, что класс Book существует с полями title, author, year, конструктором и методом printDetails() (или аналогичным для вывода).
Импортируйте необходимые пакеты: В файлах, где будете использовать Set, убедитесь, что импортированы java.util.Set и java.util.HashSet (или другая реализация). IDE подскажет, когда вы начнете писать код — используйте Ctrl+Enter для автодобавления импорта.
Выберите реализацию Set: Для этого урока используйте HashSet — это базовая реализация для уникальности без порядка. Если хотите поэкспериментировать, попробуйте LinkedHashSet для сохранения порядка вставки авторов.
Обновление класса Book (если нужно)
Класс Book уже имеет поле author типа String. Если вы не добавили геттер для author, сделайте это сейчас — он понадобится для получения имени автора при добавлении в Set.
Добавьте геттер для author:
Откройте файл Book.java.
Добавьте публичный метод getAuthor(), который возвращает значение поля author.
Это обеспечит инкапсуляцию: внешний код не будет напрямую обращаться к полю.
Создание и интеграция Set для авторов
Теперь добавим множество для уникальных авторов. Мы разместим его в классе, который управляет библиотекой, чтобы оно было общим для всех книг.
Создайте класс Library (если его нет):
В пакете проекта (или в src) щелкните правой кнопкой → New → Java Class.
Назовите класс "Library".
Этот класс будет управлять списком книг и множеством авторов.
Добавьте поле для Set:
В классе Library объявите приватное поле authors типа Set, инициализированное как new HashSet<> (или LinkedHashSet<> для порядка).
Используйте private для инкапсуляции — множество не должно быть доступно напрямую извне.
Добавьте поле для списка книг:
Если вы использовали массив Book[], перенесите его в класс Library как приватное поле (Book[] books = new Book[10]; или аналогично).
Добавьте переменную для отслеживания текущего количества книг (int bookCount = 0;), чтобы управлять заполнением массива.
Создайте метод для добавления книги:
Добавьте публичный метод addBook(Book book), который:
Проверяет, есть ли место в массиве (если bookCount < books.length).
Добавляет книгу в массив (books[bookCount] = book; bookCount++;).
Получает имя автора из книги (через getAuthor()).
Добавляет имя автора в Set authors (authors.add(authorName);).
Проверяет возвращаемое значение add(): Если true, значит автор новый; если false — автор уже существует (можно вывести сообщение, например, "Автор уже добавлен").
Добавьте метод для вывода уникальных авторов:
Создайте публичный метод printAuthors(), который перебирает Set authors (через for-each цикл) и выводит каждое имя на экран с помощью System.out.println.
Обновите класс Main для тестирования:
Откройте файл Main.java.
Создайте объект Library (Library library = new Library();).
Создайте несколько объектов Book с разными авторами (некоторые с повторяющимися, чтобы проверить уникальность).
Вызовите addBook для каждой книги.
Вызовите printAuthors() — вы должны увидеть уникальный список авторов без дубликатов.
Для проверки: Добавьте книгу с существующим автором и посмотрите, добавится ли он в Set (метод add вернет false).
#Java #для_новичков #beginner #Collections #Set
Глава 3. Set — множества
Практика: В «Библиотеке» создать коллекцию Set для хранения уникальных имён авторов.
Добавлять автора при добавлении книги, проверять уникальность.
Откройте проект «Библиотека»: Запустите IntelliJ IDEA и откройте существующий проект LibraryProject (или как вы его назвали). Убедитесь, что класс Book существует с полями title, author, year, конструктором и методом printDetails() (или аналогичным для вывода).
Импортируйте необходимые пакеты: В файлах, где будете использовать Set, убедитесь, что импортированы java.util.Set и java.util.HashSet (или другая реализация). IDE подскажет, когда вы начнете писать код — используйте Ctrl+Enter для автодобавления импорта.
Выберите реализацию Set: Для этого урока используйте HashSet — это базовая реализация для уникальности без порядка. Если хотите поэкспериментировать, попробуйте LinkedHashSet для сохранения порядка вставки авторов.
Обновление класса Book (если нужно)
Класс Book уже имеет поле author типа String. Если вы не добавили геттер для author, сделайте это сейчас — он понадобится для получения имени автора при добавлении в Set.
Добавьте геттер для author:
Откройте файл Book.java.
Добавьте публичный метод getAuthor(), который возвращает значение поля author.
Это обеспечит инкапсуляцию: внешний код не будет напрямую обращаться к полю.
Создание и интеграция Set для авторов
Теперь добавим множество для уникальных авторов. Мы разместим его в классе, который управляет библиотекой, чтобы оно было общим для всех книг.
Создайте класс Library (если его нет):
В пакете проекта (или в src) щелкните правой кнопкой → New → Java Class.
Назовите класс "Library".
Этот класс будет управлять списком книг и множеством авторов.
Добавьте поле для Set:
В классе Library объявите приватное поле authors типа Set, инициализированное как new HashSet<> (или LinkedHashSet<> для порядка).
Используйте private для инкапсуляции — множество не должно быть доступно напрямую извне.
Добавьте поле для списка книг:
Если вы использовали массив Book[], перенесите его в класс Library как приватное поле (Book[] books = new Book[10]; или аналогично).
Добавьте переменную для отслеживания текущего количества книг (int bookCount = 0;), чтобы управлять заполнением массива.
Создайте метод для добавления книги:
Добавьте публичный метод addBook(Book book), который:
Проверяет, есть ли место в массиве (если bookCount < books.length).
Добавляет книгу в массив (books[bookCount] = book; bookCount++;).
Получает имя автора из книги (через getAuthor()).
Добавляет имя автора в Set authors (authors.add(authorName);).
Проверяет возвращаемое значение add(): Если true, значит автор новый; если false — автор уже существует (можно вывести сообщение, например, "Автор уже добавлен").
Добавьте метод для вывода уникальных авторов:
Создайте публичный метод printAuthors(), который перебирает Set authors (через for-each цикл) и выводит каждое имя на экран с помощью System.out.println.
Обновите класс Main для тестирования:
Откройте файл Main.java.
Создайте объект Library (Library library = new Library();).
Создайте несколько объектов Book с разными авторами (некоторые с повторяющимися, чтобы проверить уникальность).
Вызовите addBook для каждой книги.
Вызовите printAuthors() — вы должны увидеть уникальный список авторов без дубликатов.
Для проверки: Добавьте книгу с существующим автором и посмотрите, добавится ли он в Set (метод add вернет false).
#Java #для_новичков #beginner #Collections #Set
👍3
Тестирование и отладка
После реализации протестируйте, чтобы убедиться в правильной работе уникальности.
Запустите проект:
Правой кнопкой на Main.java → Run 'Main.main()'.
В консоли увидите вывод уникальных авторов. Убедитесь, что дубликаты авторов не появляются в списке.
Проверьте уникальность:
Добавьте две книги с одним автором — в Set должно быть только одно имя.
Попробуйте добавить null как автора (если поле author позволяет) — проверьте поведение Set (HashSet позволит один null).
Отладка:
Установите breakpoint в методе addBook перед authors.add() и после — шагайте (F8) и смотрите, меняется ли размер Set.
Если ошибки: NullPointerException (если Set не инициализировано), ArrayIndexOutOfBoundsException (если массив переполнен) — добавьте проверку на размер.
В консоли выводите сообщения, например, если add вернул false: "Автор [name] уже существует".
Эксперименты:
Измените реализацию Set на LinkedHashSet — проверьте, сохраняется ли порядок добавления авторов.
Попробуйте TreeSet — добавьте Comparator, если нужно сортировать авторов по алфавиту (TreeSet требует Comparable для элементов).
Полезные советы для новичков
Инициализация Set: Всегда инициализируйте в конструкторе класса (authors = new HashSet<>();), чтобы избежать NullPointerException.
Проверка возвращаемого значения: Используйте boolean от add/remove для логики (например, уведомление о дубликате).
Custom уникальность: Если уникальность по полю (как author), убедитесь, что equals/hashCode в Book учитывают только author, если нужно (но для Set это не требуется, так как String имеет правильные методы).
Расширение проекта: Подумайте, как добавить метод для проверки, существует ли автор перед добавлением книги (используйте contains).
Производительность: Для малого количества авторов (сотни) любая реализация подойдет; для больших — HashSet fastest.
Практическое задание
Задача 1: Добавьте в Library метод isAuthorUnique(String authorName), который использует contains для проверки, есть ли автор в Set.
Задача 2: В addBook перед добавлением автора проверьте с помощью contains, и если он новый, выведите сообщение "Новый автор добавлен".
Задача 3: Создайте 5-6 книг с 3-4 уникальными авторами (некоторые повторяются), добавьте их, выведите авторов и убедитесь в уникальности.
Реализуйте эти задачи самостоятельно, следуя шагам урока. Это закрепит работу с Set в контексте проекта.
#Java #для_новичков #beginner #Collections #Set
После реализации протестируйте, чтобы убедиться в правильной работе уникальности.
Запустите проект:
Правой кнопкой на Main.java → Run 'Main.main()'.
В консоли увидите вывод уникальных авторов. Убедитесь, что дубликаты авторов не появляются в списке.
Проверьте уникальность:
Добавьте две книги с одним автором — в Set должно быть только одно имя.
Попробуйте добавить null как автора (если поле author позволяет) — проверьте поведение Set (HashSet позволит один null).
Отладка:
Установите breakpoint в методе addBook перед authors.add() и после — шагайте (F8) и смотрите, меняется ли размер Set.
Если ошибки: NullPointerException (если Set не инициализировано), ArrayIndexOutOfBoundsException (если массив переполнен) — добавьте проверку на размер.
В консоли выводите сообщения, например, если add вернул false: "Автор [name] уже существует".
Эксперименты:
Измените реализацию Set на LinkedHashSet — проверьте, сохраняется ли порядок добавления авторов.
Попробуйте TreeSet — добавьте Comparator, если нужно сортировать авторов по алфавиту (TreeSet требует Comparable для элементов).
Полезные советы для новичков
Инициализация Set: Всегда инициализируйте в конструкторе класса (authors = new HashSet<>();), чтобы избежать NullPointerException.
Проверка возвращаемого значения: Используйте boolean от add/remove для логики (например, уведомление о дубликате).
Custom уникальность: Если уникальность по полю (как author), убедитесь, что equals/hashCode в Book учитывают только author, если нужно (но для Set это не требуется, так как String имеет правильные методы).
Расширение проекта: Подумайте, как добавить метод для проверки, существует ли автор перед добавлением книги (используйте contains).
Производительность: Для малого количества авторов (сотни) любая реализация подойдет; для больших — HashSet fastest.
Практическое задание
Задача 1: Добавьте в Library метод isAuthorUnique(String authorName), который использует contains для проверки, есть ли автор в Set.
Задача 2: В addBook перед добавлением автора проверьте с помощью contains, и если он новый, выведите сообщение "Новый автор добавлен".
Задача 3: Создайте 5-6 книг с 3-4 уникальными авторами (некоторые повторяются), добавьте их, выведите авторов и убедитесь в уникальности.
Реализуйте эти задачи самостоятельно, следуя шагам урока. Это закрепит работу с Set в контексте проекта.
#Java #для_новичков #beginner #Collections #Set
👍2
Глава 2. List — списки
Метод set
Операция замены элемента в списке фундаментально отличается от операций добавления и удаления, поскольку не изменяет размер коллекции, а лишь модифицирует ее содержимое. Эта операция раскрывает компромисс между скоростью доступа к элементам и стоимостью их модификации, который по-разному разрешается в ArrayList и LinkedList. В то время как одна реализация обеспечивает практически мгновенную замену любого элемента, другая требует значительных затрат на предварительный поиск, демонстрируя тем самым trade-off между разными аспектами производительности.
ArrayList: непосредственная замена в массиве
Архитектурные предпосылки эффективной замены
ArrayList, основанный на динамическом массиве, предоставляет идеальные условия для операции замены элементов. Его внутренняя структура — непрерывный блок памяти в виде массива Object[] — позволяет осуществлять прямой доступ к любой позиции за постоянное время. Эта архитектурная особенность делает операцию set одной из наиболее эффективных операций в ArrayList.
Детальный процесс выполнения set(index, element)
Фаза валидации и проверки
Перед выполнением собственно замены элемента система осуществляет серию проверок, обеспечивающих корректность операции:
Валидация индекса:
Происходит тщательная проверка того, что указанный индекс находится в допустимом диапазоне от 0 (включительно) до текущего размера списка (исключительно). Эта проверка включает сравнение запрошенного индекса со значением поля size и при необходимости выброс исключения IndexOutOfBoundsException с детализированным сообщением.
Проверка ссылочной целостности:
Неявно обеспечивается, что внутренний массив elementData инициализирован и находится в консистентном состоянии, готовом к операции модификации.
Фаза извлечения и замены элемента
После успешной валидации начинается непосредственно процесс замены:
Прямой доступ к массиву:
Благодаря массиву как базовой структуре данных, позиция целевого элемента вычисляется как прямое смещение — для индекса i элемент находится в elementData[i].
Извлечение предыдущего значения:
Перед заменой система сохраняет ссылку на текущий элемент в указанной позиции. Это значение будет возвращено как результат операции, обеспечивая возможность отката или анализа изменений.
Непосредственная замена:
Новый элемент помещается в ту же позицию массива. Эта операция представляет собой простое присваивание ссылки в ячейке массива.
Обновление метаданных:
Несмотря на то, что размер списка не изменяется, операция set инкрементирует счетчик модификаций (modCount). Это критически важно для поддержания корректности fail-fast итераторов, которые должны обнаруживать любые структурные изменения коллекции.
Отсутствие структурных изменений
Ключевой характеристикой операции set в ArrayList является то, что она не вызывает реорганизации внутренней структуры данных. В отличие от операций add и remove, которые могут требовать расширения массива или сдвига элементов, set затрагивает только одну ячейку памяти, что делает ее исключительно легковесной.
#Java #для_новичков #beginner #List #ArrayList #LinkedList #set
Метод set
Операция замены элемента в списке фундаментально отличается от операций добавления и удаления, поскольку не изменяет размер коллекции, а лишь модифицирует ее содержимое. Эта операция раскрывает компромисс между скоростью доступа к элементам и стоимостью их модификации, который по-разному разрешается в ArrayList и LinkedList. В то время как одна реализация обеспечивает практически мгновенную замену любого элемента, другая требует значительных затрат на предварительный поиск, демонстрируя тем самым trade-off между разными аспектами производительности.
ArrayList: непосредственная замена в массиве
Архитектурные предпосылки эффективной замены
ArrayList, основанный на динамическом массиве, предоставляет идеальные условия для операции замены элементов. Его внутренняя структура — непрерывный блок памяти в виде массива Object[] — позволяет осуществлять прямой доступ к любой позиции за постоянное время. Эта архитектурная особенность делает операцию set одной из наиболее эффективных операций в ArrayList.
Детальный процесс выполнения set(index, element)
Фаза валидации и проверки
Перед выполнением собственно замены элемента система осуществляет серию проверок, обеспечивающих корректность операции:
Валидация индекса:
Происходит тщательная проверка того, что указанный индекс находится в допустимом диапазоне от 0 (включительно) до текущего размера списка (исключительно). Эта проверка включает сравнение запрошенного индекса со значением поля size и при необходимости выброс исключения IndexOutOfBoundsException с детализированным сообщением.
Проверка ссылочной целостности:
Неявно обеспечивается, что внутренний массив elementData инициализирован и находится в консистентном состоянии, готовом к операции модификации.
Фаза извлечения и замены элемента
После успешной валидации начинается непосредственно процесс замены:
Прямой доступ к массиву:
Благодаря массиву как базовой структуре данных, позиция целевого элемента вычисляется как прямое смещение — для индекса i элемент находится в elementData[i].
Извлечение предыдущего значения:
Перед заменой система сохраняет ссылку на текущий элемент в указанной позиции. Это значение будет возвращено как результат операции, обеспечивая возможность отката или анализа изменений.
Непосредственная замена:
Новый элемент помещается в ту же позицию массива. Эта операция представляет собой простое присваивание ссылки в ячейке массива.
Обновление метаданных:
Несмотря на то, что размер списка не изменяется, операция set инкрементирует счетчик модификаций (modCount). Это критически важно для поддержания корректности fail-fast итераторов, которые должны обнаруживать любые структурные изменения коллекции.
Отсутствие структурных изменений
Ключевой характеристикой операции set в ArrayList является то, что она не вызывает реорганизации внутренней структуры данных. В отличие от операций add и remove, которые могут требовать расширения массива или сдвига элементов, set затрагивает только одну ячейку памяти, что делает ее исключительно легковесной.
#Java #для_новичков #beginner #List #ArrayList #LinkedList #set
Производительность и оптимизации
Временная сложность
Операция set в ArrayList имеет временную сложность O(1) в худшем случае. Время выполнения практически идентично для замены элемента в любой позиции списка и не зависит от общего количества элементов.
Влияние на memory model
Локальность ссылок:
Поскольку операция затрагивает только одну ячейку массива, она оказывает минимальное влияние на кэширование процессора и может даже улучшить локальность, если новый элемент часто используется впоследствии.
Отсутствие аллокаций:
Операция не создает новых объектов и не требует выделения памяти, что делает ее friendly по отношению к garbage collector.
Барьеры памяти в многопоточных сценариях
При работе в многопоточной среде операция set требует proper synchronization для обеспечения visibility изменений. Присваивание ссылки в массиве само по себе является atomic операцией, но без дополнительных барьеров памяти нет гарантии, что изменение будет видно другим потокам.
LinkedList: поиск с последующей заменой
Архитектурные особенности замены в связном списке
LinkedList, реализованный как двусвязный список, подходит к операции замены элементов принципиально иным образом. Его децентрализованная структура, состоящая из отдельных узлов, распределенных в куче, требует предварительного поиска целевого узла перед выполнением собственно замены.
Структура узла и организация данных
Каждый узел LinkedList содержит три ключевых компонента, которые участвуют в операции замены:
Важно отметить, что операция set затрагивает только поле item узла, оставляя ссылки next и prev неизменными.
Детальный процесс выполнения set(index, element)
Фаза валидации и стратегического планирования
Как и в ArrayList, операция начинается с проверки корректности входных данных:
Проверка границ индекса:
Убеждаются, что индекс находится в допустимом диапазоне [0, size-1].
В зависимости от положения целевого индекса выбирается наиболее эффективная точка начала обхода:
Для индексов в первой половине списка (index < size / 2) обход начинается с головы (head)
Для индексов во второй половине обход начинается с хвоста (tail)
Эта оптимизация уменьшает среднее количество шагов поиска примерно вдвое.
Фаза поиска целевого узла
После определения начальной точки начинается процесс последовательного обхода:
Инициализация указателя обхода:
Создается временная переменная, которая устанавливается на начальный узел (head или tail).
Последовательное перемещение по цепочке:
Для каждого шага обхода:
При движении от головы указатель перемещается к node.next
При движении от хвоста указатель перемещается к node.prev
Счетчик текущей позиции инкрементируется или декрементируется соответственно
Достижение целевой позиции:
Процесс продолжается до тех пор, пока текущая позиция не совпадет с запрошенным индексом.
Фаза непосредственной замены
Когда целевой узел найден:
Сохранение предыдущего значения:
Из поля item целевого узла извлекается и сохраняется текущий элемент для последующего возврата.
Замена элемента:
В поле item целевого узла записывается ссылка на новый объект.
Обновление метаданных:
Как и в ArrayList, инкрементируется счетчик модификаций (modCount) для поддержания корректности итераторов.
Производительность и характеристики операции
Временная сложность
Операция set в LinkedList имеет временную сложность O(n) в худшем случае, где n — количество элементов в списке. Однако благодаря оптимизации двунаправленного поиска средняя сложность составляет O(n/4) = O(n).
Распределение стоимости операции
Время поиска: Составляет подавляющую часть общей стоимости операции — O(n)
Время замены: Пренебрежимо мало — O(1)
Зависимость от паттерна доступа
Худший случай: Замена элемента в середине большого списка
Лучший случай: Замена первого или последнего элемента
Средний случай: Замена элемента на расстоянии ~n/4 от ближайшего конца
#Java #для_новичков #beginner #List #ArrayList #LinkedList #set
Временная сложность
Операция set в ArrayList имеет временную сложность O(1) в худшем случае. Время выполнения практически идентично для замены элемента в любой позиции списка и не зависит от общего количества элементов.
Влияние на memory model
Локальность ссылок:
Поскольку операция затрагивает только одну ячейку массива, она оказывает минимальное влияние на кэширование процессора и может даже улучшить локальность, если новый элемент часто используется впоследствии.
Отсутствие аллокаций:
Операция не создает новых объектов и не требует выделения памяти, что делает ее friendly по отношению к garbage collector.
Барьеры памяти в многопоточных сценариях
При работе в многопоточной среде операция set требует proper synchronization для обеспечения visibility изменений. Присваивание ссылки в массиве само по себе является atomic операцией, но без дополнительных барьеров памяти нет гарантии, что изменение будет видно другим потокам.
LinkedList: поиск с последующей заменой
Архитектурные особенности замены в связном списке
LinkedList, реализованный как двусвязный список, подходит к операции замены элементов принципиально иным образом. Его децентрализованная структура, состоящая из отдельных узлов, распределенных в куче, требует предварительного поиска целевого узла перед выполнением собственно замены.
Структура узла и организация данных
Каждый узел LinkedList содержит три ключевых компонента, которые участвуют в операции замены:
Node<E> {
E item; // хранимый элемент (подлежит замене)
Node<E> next; // ссылка на следующий узел
Node<E> prev; // ссылка на предыдущий узел
}Важно отметить, что операция set затрагивает только поле item узла, оставляя ссылки next и prev неизменными.
Детальный процесс выполнения set(index, element)
Фаза валидации и стратегического планирования
Как и в ArrayList, операция начинается с проверки корректности входных данных:
Проверка границ индекса:
Убеждаются, что индекс находится в допустимом диапазоне [0, size-1].
В зависимости от положения целевого индекса выбирается наиболее эффективная точка начала обхода:
Для индексов в первой половине списка (index < size / 2) обход начинается с головы (head)
Для индексов во второй половине обход начинается с хвоста (tail)
Эта оптимизация уменьшает среднее количество шагов поиска примерно вдвое.
Фаза поиска целевого узла
После определения начальной точки начинается процесс последовательного обхода:
Инициализация указателя обхода:
Создается временная переменная, которая устанавливается на начальный узел (head или tail).
Последовательное перемещение по цепочке:
Для каждого шага обхода:
При движении от головы указатель перемещается к node.next
При движении от хвоста указатель перемещается к node.prev
Счетчик текущей позиции инкрементируется или декрементируется соответственно
Достижение целевой позиции:
Процесс продолжается до тех пор, пока текущая позиция не совпадет с запрошенным индексом.
Фаза непосредственной замены
Когда целевой узел найден:
Сохранение предыдущего значения:
Из поля item целевого узла извлекается и сохраняется текущий элемент для последующего возврата.
Замена элемента:
В поле item целевого узла записывается ссылка на новый объект.
Обновление метаданных:
Как и в ArrayList, инкрементируется счетчик модификаций (modCount) для поддержания корректности итераторов.
Производительность и характеристики операции
Временная сложность
Операция set в LinkedList имеет временную сложность O(n) в худшем случае, где n — количество элементов в списке. Однако благодаря оптимизации двунаправленного поиска средняя сложность составляет O(n/4) = O(n).
Распределение стоимости операции
Время поиска: Составляет подавляющую часть общей стоимости операции — O(n)
Время замены: Пренебрежимо мало — O(1)
Зависимость от паттерна доступа
Худший случай: Замена элемента в середине большого списка
Лучший случай: Замена первого или последнего элемента
Средний случай: Замена элемента на расстоянии ~n/4 от ближайшего конца
#Java #для_новичков #beginner #List #ArrayList #LinkedList #set
Сравнительный анализ ArrayList и LinkedList
Количественные характеристики производительности
Время выполнения:
ArrayList: 5-15 наносекунд (постоянно)
LinkedList: 10-50 наносекунд × количество пройденных узлов
Потребление памяти во время операции:
ArrayList: Не требует дополнительной памяти
LinkedList: Не требует дополнительной памяти (кроме временных переменных обхода)
Качественные различия
Локальность памяти:
ArrayList: Отличная — операция затрагивает одну ячейку в непрерывном блоке
LinkedList: Плохая — узел может находиться в произвольном месте кучи
Влияние на garbage collector:
ArrayList: Минимальное — заменяемая ссылка становится кандидатом на сборку
LinkedList: Аналогично ArrayList
Сценарии преимущественного использования
ArrayList превосходит когда:
Частые замены элементов в произвольных позициях
Критически важна предсказуемость времени выполнения
Работа с большими списками
LinkedList может быть предпочтителен когда:
Замены преимущественно происходят near концов списка
Преобладают другие операции, где LinkedList имеет преимущество
Размер списка невелик
Специализированные реализации List
CopyOnWriteArrayList
Механизм замены:
Использует стратегию "копирование при записи", что кардинально меняет семантику операции:
Создается полная копия внутреннего массива
В копии заменяется элемент в указанной позиции
Ссылка на внутренний массив атомарно заменяется на новую копию
Старый массив остается доступным для текущих читателей
Производительность:
Время выполнения: O(n) из-за необходимости копирования всего массива
Потребление памяти: Удвоенное во время операции
Thread-safe: Да, без блокировок для читателей
Vector
Устаревший synchronized подход:
Все операции, включая set, синхронизированы
Излишний overhead в single-threaded сценариях
Постоянное время доступа аналогично ArrayList
Многопоточные аспекты операции set
Проблемы конкурентного доступа
Несинхронизированные реализации:
ArrayList и LinkedList не обеспечивают thread-safe выполнение операции set:
Возможность lost updates при concurrent модификациях
Риск повреждения структур данных
Отсутствие гарантий visibility изменений
Состояние гонки:
При одновременном вызове set для одного индекса из разных потоков может сохраниться только одно из изменений.
Стратегии обеспечения потокобезопасности
Явная синхронизация:
Thread-safe обертки:
Concurrent коллекции:
Memory consistency guarantees
Для обеспечения видимости изменений между потоками необходимо установление happens-before отношений через:
Synchronized блоки
Volatile переменные
Atomic классы
Lock механизмы
Влияние на итераторы и представления
Fail-fast семантика
Операция set инкрементирует счетчик modCount, что приводит к выбросу ConcurrentModificationException при обнаружении изменения во время итерации:
Итераторы сохраняют ожидаемое значение modCount
При каждой операции итератор проверяет соответствие текущего modCount
Несоответствие приводит к немедленному исключению
Особенности ListIterator
ListIterator предоставляет собственный метод set, который имеет важные отличия:
Не инкрементирует modCount родительского списка
Может быть вызван многократно для замены текущего элемента
Более эффективен для LinkedList, так использует текущую позицию итератора
#Java #для_новичков #beginner #List #ArrayList #LinkedList #set
Количественные характеристики производительности
Время выполнения:
ArrayList: 5-15 наносекунд (постоянно)
LinkedList: 10-50 наносекунд × количество пройденных узлов
Потребление памяти во время операции:
ArrayList: Не требует дополнительной памяти
LinkedList: Не требует дополнительной памяти (кроме временных переменных обхода)
Качественные различия
Локальность памяти:
ArrayList: Отличная — операция затрагивает одну ячейку в непрерывном блоке
LinkedList: Плохая — узел может находиться в произвольном месте кучи
Влияние на garbage collector:
ArrayList: Минимальное — заменяемая ссылка становится кандидатом на сборку
LinkedList: Аналогично ArrayList
Сценарии преимущественного использования
ArrayList превосходит когда:
Частые замены элементов в произвольных позициях
Критически важна предсказуемость времени выполнения
Работа с большими списками
LinkedList может быть предпочтителен когда:
Замены преимущественно происходят near концов списка
Преобладают другие операции, где LinkedList имеет преимущество
Размер списка невелик
Специализированные реализации List
CopyOnWriteArrayList
Механизм замены:
Использует стратегию "копирование при записи", что кардинально меняет семантику операции:
Создается полная копия внутреннего массива
В копии заменяется элемент в указанной позиции
Ссылка на внутренний массив атомарно заменяется на новую копию
Старый массив остается доступным для текущих читателей
Производительность:
Время выполнения: O(n) из-за необходимости копирования всего массива
Потребление памяти: Удвоенное во время операции
Thread-safe: Да, без блокировок для читателей
Vector
Устаревший synchronized подход:
Все операции, включая set, синхронизированы
Излишний overhead в single-threaded сценариях
Постоянное время доступа аналогично ArrayList
Многопоточные аспекты операции set
Проблемы конкурентного доступа
Несинхронизированные реализации:
ArrayList и LinkedList не обеспечивают thread-safe выполнение операции set:
Возможность lost updates при concurrent модификациях
Риск повреждения структур данных
Отсутствие гарантий visibility изменений
Состояние гонки:
При одновременном вызове set для одного индекса из разных потоков может сохраниться только одно из изменений.
Стратегии обеспечения потокобезопасности
Явная синхронизация:
synchronized(list) {
list.set(index, newValue);
}Thread-safe обертки:
List<String> syncList = Collections.synchronizedList(new ArrayList<>());
syncList.set(index, newValue); // Внутренняя синхронизация
Concurrent коллекции:
CopyOnWriteArrayList<String> copyOnWriteList = new CopyOnWriteArrayList<>();
copyOnWriteList.set(index, newValue); // Atomic замена с копированием
Memory consistency guarantees
Для обеспечения видимости изменений между потоками необходимо установление happens-before отношений через:
Synchronized блоки
Volatile переменные
Atomic классы
Lock механизмы
Влияние на итераторы и представления
Fail-fast семантика
Операция set инкрементирует счетчик modCount, что приводит к выбросу ConcurrentModificationException при обнаружении изменения во время итерации:
Итераторы сохраняют ожидаемое значение modCount
При каждой операции итератор проверяет соответствие текущего modCount
Несоответствие приводит к немедленному исключению
Особенности ListIterator
ListIterator предоставляет собственный метод set, который имеет важные отличия:
Не инкрементирует modCount родительского списка
Может быть вызван многократно для замены текущего элемента
Более эффективен для LinkedList, так использует текущую позицию итератора
#Java #для_новичков #beginner #List #ArrayList #LinkedList #set