Java 迭代器Iterator的详解

为了方便的处理集合中的元素,Java中出现了一个对象,该对象提供了一些方法专门处理集合中的元素.例如删除和获取集合中的元素.该对象就叫做迭代器(Iterator).

对 Collection 进行迭代的类,称其为迭代器。还是面向对象的思想,专业对象做专业的事情,迭代器就是专门取出集合元素的对象。但是该对象比较特殊,不能直接创建对象(通过new),该对象是以内部类的形式存在于每个集合类的内部。

如何获取迭代器?Collection接口中定义了获取集合类迭代器的方法(iterator()),所以所有的Collection体系集合都可以获取自身的迭代器。


1.Iterable

正是由于每一个容器都有取出元素的功能。这些功能定义都一样,只不过实现的具体方式不同(因为每一个容器的数据结构不一样)所以对共性的取出功能进行了抽取,从而出现了Iterator接口。而每一个容器都在其内部对该接口进行了内部类的实现。也就是将取出方式的细节进行封装。

Jdk1.5之后添加的新接口, Collection的父接口. 实现了Iterable的类就是可迭代的.并且支持增强for循环。该接口只有一个方法即获取迭代器的方法iterator()可以获取每个容器自身的迭代器Iterator。(Collection)集合容器都需要获取迭代器(Iterator)于是在5.0后又进行了抽取将获取容器迭代器的iterator()方法放入到了Iterable接口中。Collection接口进程了Iterable,所以Collection体系都具备获取自身迭代器的方法,只不过每个子类集合都进行了重写(因为数据结构不同)


2.Iterator 

iterator() 返回该集合的迭代器对象

该类主要用于遍历集合对象,该类描述了遍历集合的常见方法

1java.lang. Itreable  

---| Itreable      接口 实现该接口可以使用增强for循环

        ---| Collection 描述所有集合共性的接口

             ---| List接口     可以有重复元素的集合

             ---| Set接口     不可以有重复元素的集合

public interface Iterable<T>

Itreable   该接口仅有一个方法,用于返回集合迭代器对象。

  Iterator<T> iterator() 返回集合的迭代器对象

 

Iterator接口定义的方法

Itreator	该接口是集合的迭代器接口类,定义了常见的迭代方法
	1:boolean hasNext() 
						判断集合中是否有元素,如果有元素可以迭代,就返回true。
	2: E next()  
						返回迭代的下一个元素,注意: 如果没有下一个元素时,调用
next元素会抛出NoSuchElementException

	3: void remove()
						从迭代器指向的集合中移除迭代器返回的最后一个元素(可选操
作)。

思考:为什么next方法的返回类型是Object的呢? 

为了可以接收任意类型的对象,那么返回的时候,不知道是什么类型的就定义为object

3.迭代器的遍历

第一种方式:while循环

public static void main(String[] args) {
		ArrayList list = new ArrayList();
		// 增加:add() 将指定对象存储到容器中
		list.add("计算机网络");
		list.add("现代操作系统");
		list.add("java编程思想");
		list.add("java核心技术");
		list.add("java语言程序设计");
		System.out.println(list);
		Iterator it = list.iterator();
		while (it.hasNext()) {
			String next = (String) it.next();
			System.out.println(next);
		}
	}
第二种方式:for循环

public class Demo2 {
	public static void main(String[] args) {
		ArrayList list = new ArrayList();
		// 增加:add() 将指定对象存储到容器中
		list.add("计算机网络");
		list.add("现代操作系统");
		list.add("java编程思想");
		list.add("java核心技术");
		list.add("java语言程序设计");
		System.out.println(list);

		for (Iterator it = list.iterator(); it.hasNext();) {
             //迭代器的next方法返回值类型是Object,所以要记得类型转换。
			String next = (String) it.next();
			System.out.println(next);
		}
	}
}

需要取出所有元素时,可以通过循环,java 建议使用for 循环。因为可以对内存进行一下优化。

第三种方式:使用迭代器清空集合

public class Demo1 {
	public static void main(String[] args) {
		Collection coll = new ArrayList();
		coll.add("aaa");
		coll.add("bbb");
		coll.add("ccc");
		coll.add("ddd");
		System.out.println(coll);
		Iterator it = coll.iterator();
		while (it.hasNext()) {
			it.next();
			it.remove();
		}
		System.out.println(coll);
	}
}


需要注意的细节如下:

细节一:

如果迭代器的指针已经指向了集合的末尾,那么如果再调用next()会返回NoSuchElementException异常

public class Demo2 {
	public static void main(String[] args) {
		ArrayList list = new ArrayList();
		// 增加:add() 将指定对象存储到容器中
		list.add("计算机网络");
		list.add("现代操作系统");
		list.add("java编程思想");
		list.add("java核心技术");
		list.add("java语言程序设计");
		System.out.println(list);

		Iterator it = list.iterator();
		while (it.hasNext()) {
			String next = (String) it.next();
			System.out.println(next);
		}
		// 迭代器的指针已经指向了集合的末尾
		// String next = (String) it.next();
		// java.util.NoSuchElementException
	}
}

细节二:

 如果调用remove之前没有调用next是不合法的,会抛出IllegalStateException

public class Demo2 {
	public static void main(String[] args) {
		ArrayList list = new ArrayList();
		// 增加:add() 将指定对象存储到容器中
		list.add("计算机网络");
		list.add("现代操作系统");
		list.add("java编程思想");
		list.add("java核心技术");
		list.add("java语言程序设计");
		System.out.println(list);

		Iterator it = list.iterator();
		while (it.hasNext()) {
			// 调用remove之前没有调用next是不合法的
			// it.remove();
			// java.lang.IllegalStateException
			String next = (String) it.next();
			System.out.println(next);
		}

	}
}
4.迭代器的原理

查看ArrayList源码

private class Itr implements Iterator<E> {

		int cursor = 0;

		int lastRet = -1;

		int expectedModCount = modCount;

		public boolean hasNext() {
			return cursor != size();
		}

		public E next() {
			checkForComodification();
			try {
				E next = get(cursor);
				lastRet = cursor++;
				return next;
			} catch (IndexOutOfBoundsException e) {
				checkForComodification();
				throw new NoSuchElementException();
			}
		}

		public void remove() {
			if (lastRet == -1)
				throw new IllegalStateException();
			checkForComodification();

			try {
				AbstractList.this.remove(lastRet);
				if (lastRet < cursor)
					cursor--;
				lastRet = -1;
				expectedModCount = modCount;
			} catch (IndexOutOfBoundsException e) {
				throw new ConcurrentModificationException();
			}
		}

		
	}

注意:1.在对集合进行迭代过程中,不允许出现迭代器以外的对元素的操作,因为这样会产生安全隐患,java会抛出异常并发修改异常(ConcurrentModificationException),普通迭代器只支持在迭代过程中的删除动作。

     2.ConcurrentModificationException: 当一个集合在循环中即使用引用变量操作集合又使用迭代器操作集合对象, 会抛出该异常。

public class Demo1 {
	public static void main(String[] args) {
		Collection coll = new ArrayList();
		coll.add("aaa");
		coll.add("bbb");
		coll.add("ccc");
		coll.add("ddd");
		System.out.println(coll);
		Iterator it = coll.iterator();
		while (it.hasNext()) {
			it.next();
			it.remove();
			coll.add("abc"); // 出现了迭代器以外的对元素的操作
		}
		System.out.println(coll);
	}
}

如果是List集合,想要在迭代中操作元素可以使用List集合的特有迭代器ListIterator,该迭代器支持在迭代过程中,添加元素和修改元素。

5.List特有的迭代器ListIterator

public interface ListIterator extends Iterator

ListIterator<E> listIterator()

---| Iterator
		hasNext()
		next()
		remove()
	   ------| ListIterator Iterator子接口 List专属的迭代器
                  add(E e)    将指定的元素插入列表(可选操作)。该元素直接插入到 next 返回的下一个元素的前面(如果有)
                  void set(E o)   用指定元素替换 next 或 previous 返回的最后一个元素
                  hasPrevious()    逆向遍历列表,列表迭代器有多个元素,则返回 true。
                  previous()       返回列表中的前一个元素。

Iterator在迭代时,只能对元素进行获取(next())和删除(remove())的操作。

对于 Iterator 的子接口ListIterator 在迭代list 集合时,还可以对元素进行添加

(add(obj)),修改set(obj)的操作。

public class Demo2 {
	public static void main(String[] args) {
		ArrayList list = new ArrayList();
		// 增加:add() 将指定对象存储到容器中
		list.add("计算机网络");
		list.add("现代操作系统");
		list.add("java编程思想");
		list.add("java核心技术");
		list.add("java语言程序设计");
		System.out.println(list);
         // 获取List专属的迭代器
		ListIterator lit = list.listIterator();

		while (lit.hasNext()) {
			String next = (String) lit.next();
			System.out.println(next);
		}

	}
}

倒序遍历

public class Demo2 {
	public static void main(String[] args) {
		ArrayList list = new ArrayList();
		// 增加:add() 将指定对象存储到容器中
		list.add("计算机网络");
		list.add("现代操作系统");
		list.add("java编程思想");
		list.add("java核心技术");
		list.add("java语言程序设计");
		System.out.println(list);
        // 获取List专属的迭代器
		ListIterator lit = list.listIterator();
		while (lit.hasNext()) {
			String next = (String) lit.next();
			System.out.println(next);
		}
		System.out.println("***************");
		while (lit.hasPrevious()) {
			String next = (String) lit.previous();
			System.out.println(next);
		}

	}
}

Set方法:用指定元素替换 next 或 previous 返回的最后一个元素

public class Demo2 {
	public static void main(String[] args) {
		ArrayList list = new ArrayList();
		// 增加:add() 将指定对象存储到容器中
		list.add("计算机网络");
		list.add("现代操作系统");
		list.add("java编程思想");
		list.add("java核心技术");
		list.add("java语言程序设计");
		System.out.println(list);

		ListIterator lit = list.listIterator();
		lit.next(); // 计算机网络
		lit.next(); // 现代操作系统
		System.out.println(lit.next()); // java编程思想
		//用指定元素替换 next 或 previous 返回的最后一个元素
		lit.set("平凡的世界");// 将java编程思想替换为平凡的世界
		System.out.println(list);

	}
}

add方法将指定的元素插入列表,该元素直接插入到 next 返回的元素的后面

public class Demo2 {
	public static void main(String[] args) {
		ArrayList list = new ArrayList();
		// 增加:add() 将指定对象存储到容器中
		list.add("计算机网络");
		list.add("现代操作系统");
		list.add("java编程思想");
		list.add("java核心技术");
		list.add("java语言程序设计");
		System.out.println(list);

		ListIterator lit = list.listIterator();
		lit.next(); // 计算机网络
		lit.next(); // 现代操作系统
		System.out.println(lit.next()); // java编程思想
		// 将指定的元素插入列表,该元素直接插入到 next 返回的元素的后
		lit.add("平凡的世界");// 在java编程思想后添加平凡的世界
		System.out.println(list);

	}
}

  • 73
    点赞
  • 308
    收藏
    觉得还不错? 一键收藏
  • 12
    评论
### 回答1: Java中的迭代器Iterator)是一种对象,它可以遍历一个容器中的元素。迭代器模式是一种设计模式,可以将容器的遍历和实现分离。 Java中的迭代器Iterator)提供了一种方便的方法来遍历容器中的元素。它提供了三个基本操作: - hasNext(): 返回容器中是否还有下一个元素 - next(): 返回容器中的下一个元素 - remove(): 删除容器中最后一个被next()返回的元素 Java中的迭代器接口是Iterator,其中包含上述三个方法,实现类可以自定义实现这些方法以实现遍历。常见的容器类如List、Set都提供了迭代器的实现。使用迭代器可以更方便地遍历容器中的元素,并且可以在遍历过程中删除元素而不影响其他遍历操作。 ### 回答2: Java迭代器Iterator)是Java集合框架中一个非常重要的接口,用于遍历并访问集合中的元素。通过使用迭代器,我们可以避免使用传统的for循环方式遍历集合,从而使代码更加简洁、优雅和易于维护。 Java迭代器接口定义了一些常用的方法,使得我们可以在不管具体集合的实现细节的情况下,访问和处理集合中的元素。常用的方法包括: 1. hasNext(): 判断集合是否还有下一个元素。 2. next(): 返回集合中的下一个元素,并将迭代器指针向下一个元素移动。 3. remove(): 移除集合中的当前元素。 下面我们来看一个简单的例子,用来演示如何使用Java迭代器进行遍历: ``` ArrayList<String> list = new ArrayList<String>(); list.add("Java"); list.add("Python"); list.add("Ruby"); Iterator<String> it = list.iterator(); while (it.hasNext()) { String element = it.next(); System.out.println(element); } ``` 在上面的例子中,我们首先定义了一组字符串元素,并将它们添加到了一个ArrayList集合中。然后通过调用iterator()方法,得到了该集合的迭代器。接着,通过使用while循环和next()方法,逐个遍历集合中的元素,并将元素内容输出到控制台上。 需要注意的是,当调用next()方法时,如果集合中已经没有更多的元素可供遍历了,则会抛出NoSuchElementException异常。因此,在使用Java迭代器时,必须首先使用hasNext()方法来检查集合是否还有下一个元素可供遍历。 还需要注意的是,除了一些具体的集合类,如ArrayList,LinkedList, HashSet, TreeMap等,Java迭代器还可以用于自定义集合类。只要实现了Iterator接口,并在实现中定义next()、hasNext()、remove()三个方法即可。 总之,Java迭代器Java集合框架中一项非常重要的功能。通过使用迭代器,我们可以遍历和访问集合中的元素,避免了使用传统的for循环方式,从而使代码更加简洁、优雅和易于维护。 ### 回答3: Java中的迭代器Iterator)是一种用于遍历集合框架(Collection Framework)中元素的标准方式。Iterator返回的是一个迭代器对象,该对象允许程序员处理集合框架中储存的元素序列。 要使用Iterator遍历一个集合框架中的元素,需要按照以下步骤进行: 1. 调用集合框架中的iterator()方法,返回一个Iterator类型的对象; 2. 使用next()方法取出序列中的每一个元素,next()方法返回序列中的下一个元素,直到序列中的所有元素都被访问完; 3. 使用hasNext()方法判断序列中是否还有下一个元素,如果还有,则返回true,否则返回false。 另外,Iterator类型的对象只能单向遍历,即只能使用next()方法往后遍历,无法往前遍历。 Iterator除了提供基本的遍历操作外,还提供了一些额外的可选操作,例如: 1. remove()方法:从集合中移除迭代器返回的最后一个元素(如果支持的话); 2. forEachRemaining()方法:对集合中的每个元素执行指定的操作,直到所有元素都已被处理完毕或操作引发异常。 在使用Iterator时,需要注意以下几个点: 1. 迭代器只能在对应的集合框架中进行遍历操作,否则会抛出ConcurrentModificationException异常; 2. 迭代器是单向的,一旦遍历到末尾,就无法回到开头; 3. 当集合中的元素被修改、插入或删除时,遍历过程就会中断并抛出异常; 4. 迭代器虽然是fail-fast的,但这并不意味着它一定是线程安全的,如果多个线程同时访问同一个集合,就需要进行加锁、同步等操作,否则可能会出现不可预期的结果。 总的来说,Iterator是一个十分重要的概念,是Java中集合框架遍历元素的标准方式。使用Iterator可以很方便地遍历集合中的元素,并且提供了一些可选操作,但在使用时需要注意迭代器的特殊性质,以免出现异常情况。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值