jdk1.8_lambda

参考Haskell-函数式编程

核心功能性函数接口(有且只有一个抽象方法的接口)

  • Consumer:消费型接口 void acept(T t);
    1
    2
    3
    4
    //消费型接口Consumer,输入一个参数,对其进行打印输出
    Consumer<String> consumer = (x) -> System.out.println(x);
    //打印字符串
    consumer.accept("Hello world!");
  • Supplier:供给型接口 T get();
    1
    2
    3
    4
    //供给型接口Supplier,返回指定字符串
    Supplier<String> supplier = () -> "Hello world!";
    //获取字符串
    supplier.get();
  • Function<T,R> :函数型接口 R apply(T t);
    1
    2
    3
    Function<String, Integer> function = (x) -> x.length();
    //获取字符串长度
    function.apply("Hello world!");
  • Predicate:断言型接口 boolean test(T t);
    1
    2
    3
    4
    //断言型接口Predicate,输入数字,判断是否大于0
    Predicate<Integer> predicate = (x) -> x > 0;
    //获取判断结果
    predicate.test(10);

案例

1.Iterable.forEach(Consumer<? super T> action)

1
2
3
4
5
6
7
8
9
List<String> soutList = Arrays.asList("one","two","three");
soutList.forEach(out-> System.out.println(out));
soutList.forEach(System.out::println);

Page p = new Page("content");
Page p = Page.new("content");
public static Stream<Page> createPagesFrom(Stream<String>; contents) {
return contents.map(Page::new).
}

Iterable接口
forEach方法

1). Iterable接口允许一个对象成为forEach的目标

2). 这里的forEach被声明为一个默认方法,并接收了一个消费者类型的函数接口,先查找匹配对应的操作,若不为空则执行操作

3). 接口中的默认方法可以不被实现类实现

4). 如上代码中,输出集合soutList调用了foreach方法,箭头左边是参数,右边是抽象方法的实现内容或已有方法的引用

5). 当参数列表中的参数和方法签名匹配时,可以通过 :: 直接引用方法,包括构造方法

2.Collection.stream()、Stream、Stream#filter(Predicate)、Collector、Collectors、Collectors#toList()

1
2
3
4
5
6
7
8
9
10
11
12
13
14
  /**
* Complete this for Exercise_2_Test#getAllPersonsEligibleToVote
*
* 根据给定年龄返回给定选民集合中合法年龄的选民集合
* @param potentialVoters - voters to filter
* @param legalAgeOfVoting - age where it's legal to vote
* @return a list of eligible voters
*/
public static List<Person> eligibleVoters(List<Person> potentialVoters, int legalAgeOfVoting) {
Stream<Person> personStream = potentialVoters.stream().filter(
person -> person.getAge()>=legalAgeOfVoting
);
return personStream.collect(Collectors.toList());
}

1). Collection#stream() 返回一个Stream序列

2). Stream 流

- A sequence of elements supporting sequential and parallel aggregate operations
- 支撑顺序和并行聚合操作元素的序列
- lazy: computation on the source data is only performed when the terminal operation is initiated, and source elements are consumed only as needed.
- 惰式执行:即仅在终端操作启动时才执行对源数据的计算,并且仅在需要时使用源元素。
- 在对流进行“操作”(类似查询和过滤)时,这些行为应该是无干扰(不修改流的数据源)且无状态(其结果不依赖于流管道执行过程中可能改变的任何状态)的
- 消费性质,只使用一次,不可以被重用

3). Stream#filter(Predicate) 对流进行过滤 中间操作

4). Predicate 断言功能性函数接口,返回Boolean值 见上文

5). Stream.collect()

- 返回一个结果容器 
- <R, A> R collect(Collector<? super T, A, R> collector)
- <R> 结果的类型
- <A> Collector的积累类型

6). Collector 执行结果转换 Collectors 实现Collector的各种还原操作

- public interface Collector<T, A, R>
- <T>  输入元素的类型
- <A>  还原运算的可变累积型(通常隐藏作为一个实现细节)
- <R>  还原运算的结果类

7). Collectors#toList() 返回一个Collector ,其累积的输入元素融入到一个新的List中

- public static <T> Collector<T, ?, List<T>> toList()
- <T>  输入元素的类型

3.Stream#map(Function)

1
2
3
4
5
6
7
8
9
10
  /**
* Apply a mapping of Books to titles (Strings)
*
* 将书籍的标题映射到一个新的List<String>
* @param books - books to transform
* @return list of book titles
*/
public static List<String> titlesOf(List<Book> books) {
return books.stream().map(Book::getTitle).collect(Collectors.toList());
}

1). Stream#map(Function)

- 返回指定类型的流 中间操作
- Function 函数型功能接口 见上文
- java.util.stream.Stream<T> <R> Stream<R> map(Function<? super T, ? extends R> mapper)
- <? super T>/<? extends T> <? super T>表示包括T在内的任何T的父类,<? extends T>表示包括T在内的任何T的子类

reduce

U reduce(U identity,BiFunction<U, ? super T, U> accumulator,BinaryOperator combiner)

在串行流(stream)中,第三个参数combiner不会起作用。

在并行流(parallelStream)中,流被fork join出多个线程进行执行,此时每个线程的执行流程就跟第二个方法reduce(identity, accumulator)一样,而第三个参数combiner函数,则是将每个线程的执行结果当成一个新的流,然后使用第一个方法reduce(accumulator)流程进行规约。

Listnumbers = Arrays.asList(1, 2, 3, 4, 5);

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
// 示例1
System.out.println( numbers.stream().reduce(Integer::sum).get() );
System.out.println( numbers.stream().reduce(100, Integer::sum) );
System.out.println( numbers.stream().reduce(100, (x,y)-> x = x+y, (x,y)-> x = x+y) );
System.out.println( numbers.parallelStream().reduce(100, (x,y)-> x = x+y, (x,y)-> x = x+y) );
//15
//115
//115
//515

// 示例2
List<Integer> list = new ArrayList<>();
list.add(100);
System.out.println(
numbers.stream().reduce(list,
(x,y)-> {List<Integer> ll = new ArrayList<>(x);ll.add(y); return ll;},
(x,y)-> {List<Integer> ll = new ArrayList<>(x);ll.addAll(y); return ll;})
);
System.out.println(
numbers.parallelStream().reduce(list,
(x,y)-> {List<Integer> ll = new ArrayList<>(x);ll.add(y); return ll;},
(x,y)-> {List<Integer> ll = new ArrayList<>(x);ll.addAll(y); return ll;})
);
//[100, 1, 2, 3, 4, 5]
//[100, 1, 100, 2, 100, 3, 100, 4, 100, 5]