В интерфейсе iterator представлены всего три метода:
□ логический метод hasNext () возвращает true, если обход еще не завершен;
□ метод next () делает текущим следующий элемент коллекции и возвращает его в виде объекта класса Object;
□ метод remove () удаляет текущий элемент коллекции.
Можно представить себе дело так, что итератор — это указатель на элемент коллекции. При создании итератора указатель устанавливается перед первым элементом, метод next () перемещает указатель на первый элемент и показывает его. Следующее применение метода next () перемещает указатель на второй элемент коллекции и демонстрирует его. Последнее применение метода next () выводит указатель за последний элемент коллекции.
Метод remove () позволяет при просмотре коллекции удалять из нее ненужные элементы, сохраняя при этом порядок следования элементов. Дело в том, что метод remove () самой коллекции, удалив элемент, перестроит оставшиеся элементы коллекции и итератор может неправильно просмотреть оставшуюся часть коллекции.
В листинге 6.5 к тексту листинга 6.1 добавлена работа с итератором. Впрочем, для обхода коллекции типа List можно использовать оператор "for-each". Этот способ тоже показан в листинге 6.5.
Листинг 6.5. Использование итератора вектораj
Vector v = new Vector<>();
String s = "Строка, которую мы хотим разобрать на слова.";
StringTokenizer st = new StringTokenizer(s, " \t\n\r,.");
while (st.hasMoreTokens()){
// Получаем слово и заносим в вектор
v.add(st.nextToken()); // Добавляем элемент в конец вектора.
}
System.out.println(v.firstElement()); // Первый элемент. System.out.println(v.lastElement()); // Последний элемент. v.setSize(4); // Уменьшаем число элементов.
v.add("собрать."); // Добавляем в конец укороченного вектора.
v.set(3, "опять"); // Ставим элемент в позицию 3.
// Первый способ обхода коллекции типа List // использует индексы ее элементов:
for (int i = 0; i < v.size(); i++) // Перебираем весь вектор.
System.out.print(v.get(i) + " ");
System.out.println();
// Второй способ обхода коллекции использует итератор:
Iterator it = v.iterator(); // Получаем итератор вектора.
try{
while (it.hasNext()) // Пока в векторе есть элементы,
System.out.println(it.next()); // выводим текущий элемент.
}catch(Exception e){}
// Третий способ обхода коллекции использует оператор for-each: for (String s: v) // Цикл по всем элементам вектора.
System.out.println(s); // Выводим текущий элемент вектора.
Интерфейс ListIterator
Интерфейс Listiterator расширяет интерфейс iterator, обеспечивая перемещение по коллекции как в прямом, так и в обратном направлении. Он может быть реализован только в тех коллекциях, в которых есть понятия следующего и предыдущего элемента и где элементы пронумерованы.
В интерфейс Listiterator добавлены следующие методы:
□ void add (Obj ect element) — добавляет элемент element перед текущим элементом;
□ boolean hasPrevious() — возвращает true, если в коллекции есть элементы, стоящие перед текущим элементом;
□ int nextindex () — возвращает индекс текущего элемента; если текущим является последний элемент коллекции, возвращает размер коллекции;
□ Object previous () — возвращает предыдущий элемент и делает его текущим;
□ int previousindex() — возвращает индекс предыдущего элемента;
□ void set (Obj ect element) заменяет текущий элемент элементом element; выполняется
сразу после next () или previous ( ).
Как видите, итераторы могут изменять коллекцию, в которой они работают, добавляя, удаляя и заменяя элементы. Чтобы это не приводило к конфликтам, предусмотрена исключительная ситуация, возникающая при попытке использования итераторов параллельно "родным" методам коллекции. Именно поэтому в листинге 6.5 действия с итератором заключены в блок try{} catch () {}.
Изменим часть листинга 6.5 с использованием итератора Listiterator.
// Текст листинга 6.1...
// ...
ListIterator lit = v.listIterator(); // Получаем итератор вектора.
// Указатель сейчас находится перед началом вектора.
try{
while(lit.hasNext()) // Пока в векторе есть элементы,
System.out.println(lit.next()); // переходим к следующему
// элементу и выводим его.
// Теперь указатель находится за концом вектора.
// Перейдем к началу вектора. while(lit.hasPrevious())
System.out.println(lit.previous());
}catch(Exception e){}
Интересно, что повторное применение методов next () и previous () друг за другом будет выдавать один и тот же текущий элемент.
Читать дальше
Конец ознакомительного отрывка
Купить книгу