list按位置对元素排序,元素x在列表中的位置,也称为其索引。一个列表允许重复的元素。list通过其索引来区分相同的对象。
set是无序无重复集合。与列表不同,set没有其元素的位置概念。集合的实现通常针对搜索进行了优化,所以可以快速找到特定的存储值。
map是一个集合,它根据x的一个属性来存储对象x。这意味着映射中的每个元素都是一个键-值对。键是对象的属性,而值是对象本身。映射允许根据其键快速检索一个值。
JCF Iterable
我们认为一个lambda表达式是一个对象,就和根据person类创建一个对象一样。那么一个lambda表达式对象的类型怎么确定呢,它属于哪个类呢?
我们定义了一系列接口,虽然接口不能作为一个对象的类名,但是这里我们暂且这么认为,既我们定义的这一系列不同类型的接口对应了不同lambda表达式对象的类型。
The generic interface Functiontakes a parameter of type T and returns a value of type R . A bi-function takes two parameters and returns a value: the generic interface BiFunction takes two parameters of type T and U , in that order, and returns a value of type R .It is useful to have a short hand notation for functional types. We will use the notation T→R for the type Functionand (T, U)→R for the type BiFunction . Similar shorthand notation will be used for other functional types.There are also functional interfaces that involve the primitive types int , long , and double . For example, the interface IntFunction is the type int →R, the interface LongFunction is the type long →R, and IntToLongFunction is the type int → long .An operator is a function whose parameters and return value have the same type. A binary operator is an operator that takes two parameters, and a unary operator is an operator that takes a single parameter.Thus BinaryOperatorhas type (T, T)→T, the interface UnaryOperator has type T→T, and LongUnaryOperator has type long → long . A consumer takes at least one parameter and returns no result. The interface Consumerhas type T→ void , while the interface BiConsumer has type (T, U)→ void . A supplier is the opposite of a consumer: it takes no parameters and returns a value. The interface Supplierhas type ( )→T and IntSupplier has type ( )→ int . A predicate is a function that returns a boolean value. Thus Predicatehas type T→ boolean , while BiPredicate has type (T, U)→ boolean .
可以看到,我们根据传入参数和传出参数的数量,定义了不同lambda表达式的类型。
Iterable
void forEach(Consumer super E> action)
这个方法的语义是:forEach方法接受一个Consumer super E> action类型的变量,根据刚刚的知识我们知道这个类型的变量是一个lambda表达式,而这个lambda表达式接受一个是E的超类的参数作为lambda表达式的输入,且不返回值。E是接口Iterable
假设我们有一个 `List
List list = Arrays.asList(1, 2, 3, 4, 5);
list.forEach((Integer n) -> System.out.println(n));
在这个例子中,lambda 表达式【(Integer n) -> System.out.println(n)】的类型是 `Consumer super Integer>`,这个lambda表达式接受一个interger类超类的参数作为其参数。
注意,由于 Java 8 的类型推断机制,我们不需要显式地指定参数类型。上面的代码可以简写为:
List list = Arrays.asList(1, 2, 3, 4, 5);
list.forEach(n -> System.out.println(n));
在这个简写的版本中,编译器会根据上下文推断出 lambda 表达式的类型。
在刚刚的例子中,我们发现for each方法可以对一个集合中的所有元素做相同操作,但是如果我希望对集合中的某些特定元素做不同操作该怎么办呢?
Iterator iterator()
调用这个方法,我们就对集合创建了一个iterator对象(和之前的forEach不同,foreach是对集合本身操作,这个是根据集合本身,创建一个新对象,对新对象操作),这个对象除了包含集合中的内容之外,还包含了对这些新对象的一些操作方法,我们可以对集合中的元素进行精确操作,其包含的方法如下:
例子:
import java.util.ArrayList;
import java.util.Iterator;
import java.util.function.Consumer;
/**
* This program demonstrates Iterators.
*/
public class IteratorDemo {
public static void main(String[] args) {
// Array of names.
String[] names = {"Anna", "Bob", "Carlos", "Debby"};
// Create list and add names.
ArrayList nameList = new ArrayList<>();
for (String name : names) {
nameList.add(name);
}
// Define an action for the "remaining" elements.
Consumer action = x -> {
System.out.printf("%s\n", x);
};
// Get the iterator to the list.
Iterator iter = nameList.iterator();
// Process list elements with the iterator.
while (iter.hasNext()) {
String name = iter.next();
System.out.printf("%s %d\n", name, name.length());
if (name.equals("Bob")) {
// Act differently for names after "Bob".
iter.forEachRemaining(action);
}
}
}
}
Anna 4Bob 3CarlosDebby
这里我们发现,对于bob之后的人,我们做了不一样的操作 。
特别的,我们看一个removeIf的例子:
import java.util.ArrayList;
import java.util.function.Predicate;
/**
* This program demonstrates the Collection removeIf.
*/
public class FilterDemo {
public static void main(String[] args) {
// Array of names.
String[] names = {"Anna", "Bob", "Carlos", "Debby"};
// Create list and add names.
ArrayList nameList = new ArrayList<>();
for (String name : names) {
nameList.add(name);
}
// Use forEach with lambda expression to print.
Predicate filter = x -> x.length() <= 4;
// Remove strings with length at most 4 from nameList
nameList.removeIf(filter);
// Print the array list to show remaining strings.
System.out.println(nameList);
}
}
这个例子中,我们用lambda表达式创建了一个检查谓词(predicate),将这个lambda表达式作为一个参数传入removeIf函数。