如果我有一个包含[alice, bob, abigail, charlie]
的列表,并且我想编写一个迭代器,以便迭代以'a'开头的元素,我可以编写自己的吗?我怎样才能做到这一点 ?
答案 0 :(得分:177)
最好的可重用选项是实现Iterable接口并覆盖方法iterator()。
这是一个类似ArrayList的实现接口的示例,在该类中重写Iterator()方法。
import java.util.Iterator;
public class SOList<Type> implements Iterable<Type> {
private Type[] arrayList;
private int currentSize;
public SOList(Type[] newArray) {
this.arrayList = newArray;
this.currentSize = arrayList.length;
}
@Override
public Iterator<Type> iterator() {
Iterator<Type> it = new Iterator<Type>() {
private int currentIndex = 0;
@Override
public boolean hasNext() {
return currentIndex < currentSize && arrayList[currentIndex] != null;
}
@Override
public Type next() {
return arrayList[currentIndex++];
}
@Override
public void remove() {
throw new UnsupportedOperationException();
}
};
return it;
}
}
此类使用Generics实现Iterable接口。考虑到你有数组的元素,你将能够获得一个Iterator的实例,例如,“foreach”循环所使用的实例。
您可以创建迭代器的匿名实例,而无需创建扩展Iterator并利用currentSize的值来验证您可以在数组中导航的位置(假设您创建了一个容量为10的数组,但是您在0和1处只有2个元素。该实例将拥有其所有者的所有者计数器,您需要做的就是使用hasNext()来验证当前值是否为null,以及next(),它将返回currentIndex的实例。以下是使用此API的示例...
public static void main(String[] args) {
// create an array of type Integer
Integer[] numbers = new Integer[]{1, 2, 3, 4, 5};
// create your list and hold the values.
SOList<Integer> stackOverflowList = new SOList<Integer>(numbers);
// Since our class SOList is an instance of Iterable, then we can use it on a foreach loop
for(Integer num : stackOverflowList) {
System.out.print(num);
}
// creating an array of Strings
String[] languages = new String[]{"C", "C++", "Java", "Python", "Scala"};
// create your list and hold the values using the same list implementation.
SOList<String> languagesList = new SOList<String>(languages);
System.out.println("");
// Since our class SOList is an instance of Iterable, then we can use it on a foreach loop
for(String lang : languagesList) {
System.out.println(lang);
}
}
// will print "12345
//C
//C++
//Java
//Python
//Scala
如果需要,您也可以使用Iterator实例迭代它:
// navigating the iterator
while (allNumbers.hasNext()) {
Integer value = allNumbers.next();
if (allNumbers.hasNext()) {
System.out.print(value + ", ");
} else {
System.out.print(value);
}
}
// will print 1, 2, 3, 4, 5
foreach文档位于http://download.oracle.com/javase/1,5.0/docs/guide/language/foreach.html。您可以在我的个人练习中google code查看更完整的实施。
现在,为了获得你需要的效果,我认为你需要在Iterator中插入一个过滤器的概念......由于迭代器依赖于下一个值,因此很难在hasNext()上返回true ,然后使用不以char“a”开头的值过滤next()实现。我认为您需要使用具有给定过滤器的值的过滤列表来使用辅助Interator。
答案 1 :(得分:40)
不确定。迭代器只是java.util.Iterator
接口的一个实现。如果您正在使用来自LinkedList
的现有可迭代对象(例如,java.util
),则需要对其进行子类化并覆盖其iterator
函数,以便您返回自己的,或提供在特殊Iterator
实例中包装标准迭代器的方法(具有更广泛使用的优点)等。
答案 2 :(得分:10)
Iterable计算阶乘
的好例子FactorialIterable fi = new FactorialIterable(10);
Iterator<Integer> iterator = fi.iterator();
while (iterator.hasNext()){
System.out.println(iterator.next());
}
Java 1.8的简短代码
new FactorialIterable(5).forEach(System.out::println);
自定义Iterable类
public class FactorialIterable implements Iterable<Integer> {
private final FactorialIteartor factorialIteartor;
public FactorialIterable(Integer value) {
factorialIteartor = new FactorialIteartor(value);
}
@Override
public Iterator<Integer> iterator() {
return factorialIteartor;
}
@Override
public void forEach(Consumer<? super Integer> action) {
Objects.requireNonNull(action);
Integer last = 0;
for (Integer t : this) {
last = t;
}
action.accept(last);
}
}
自定义迭代器类
public class FactorialIteartor implements Iterator<Integer> {
private final Integer mNumber;
private Integer mPosition;
private Integer mFactorial;
public FactorialIteartor(Integer number) {
this.mNumber = number;
this.mPosition = 1;
this.mFactorial = 1;
}
@Override
public boolean hasNext() {
return mPosition <= mNumber;
}
@Override
public Integer next() {
if (!hasNext())
return 0;
mFactorial = mFactorial * mPosition;
mPosition++;
return mFactorial;
}
}
答案 3 :(得分:6)
这是编写迭代器的完整代码,它迭代以'a'开头的元素:
import java.util.Iterator;
public class AppDemo {
public static void main(String args[]) {
Bag<String> bag1 = new Bag<>();
bag1.add("alice");
bag1.add("bob");
bag1.add("abigail");
bag1.add("charlie");
for (Iterator<String> it1 = bag1.iterator(); it1.hasNext();) {
String s = it1.next();
if (s != null)
System.out.println(s);
}
}
}
自定义迭代器类
import java.util.ArrayList;
import java.util.Iterator;
public class Bag<T> {
private ArrayList<T> data;
public Bag() {
data = new ArrayList<>();
}
public void add(T e) {
data.add(e);
}
public Iterator<T> iterator() {
return new BagIterator();
}
public class BagIterator<T> implements Iterator<T> {
private int index;
private String str;
public BagIterator() {
index = 0;
}
@Override
public boolean hasNext() {
return index < data.size();
}
@Override
public T next() {
str = (String) data.get(index);
if (str.startsWith("a"))
return (T) data.get(index++);
index++;
return null;
}
}
}
答案 4 :(得分:4)
您可以实现自己的迭代器。您的迭代器可以构造为包装List返回的Iterator,或者您可以保留游标并使用List的get(int index)方法。您只需要为Iterator的下一个方法添加逻辑,并使用hasNext方法来考虑您的过滤条件。您还必须决定迭代器是否支持删除操作。
答案 5 :(得分:1)
这是问题的完整答案。
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
class ListIterator implements Iterator<String>{
List<String> list;
int pos = 0;
public ListIterator(List<String> list) {
this.list = list;
}
@Override
public boolean hasNext() {
while(pos < list.size()){
if (list.get(pos).startsWith("a"))
return true;
pos++;
}
return false;
}
@Override
public String next() {
if (hasNext())
return list.get(pos++);
throw new NoSuchElementException();
}
}
public class IteratorTest {
public static void main(String[] args) {
List<String> list = Arrays.asList("alice", "bob", "abigail", "charlie");
ListIterator itr = new ListIterator(list);
while(itr.hasNext())
System.out.println(itr.next()); // prints alice, abigail
}
}
ListIterator
是数组的迭代器,该数组返回以'a'开头的元素。 NoSuchElementException
异常。