java 函数式接口Consumer源码分析

目录

简介

accept方法

andThen方法

经典应用-iterable

扩展类介绍


简介

Consumer是一个函数式接口,它有一个需要覆盖的方法accept,代表接受一个参数输入且没有任何返回值的操作。不同于其它的函数式接口,Consumer期望通过方法的实现来执行具体的操作

accept方法

可以看到它接受一个参数,什么都不返还

    /**
     * Performs this operation on the given argument.
     * 

接受一个参数,覆盖的方法对它做出动作,什么都不返回 * @param t the input argument */ void accept(T t);

具体使用例子

		Consumer consumer=new Consumer() {
			@Override
	        public void accept(String t) {
	                System.out.println(t+"123");
	        }
		};
		consumer.accept("abc");

最后打印abc123

andThen方法

这个方法返回一个组合过的consumer实例,
它的accept方法是先调用this这个consumer的accept方法,然后调用after的accept方法

    /**
     * Returns a composed {@code Consumer} that performs, in sequence, this
     * operation followed by the {@code after} operation. If performing either
     * operation throws an exception, it is relayed to the caller of the
     * composed operation.  If performing this operation throws an exception,
     * the {@code after} operation will not be performed.
     * 

返回一个组合过的consumer实例, *

它的accept方法是先调用this这个consumer的accept方法,然后调用after的accept方法 *

被抛出的异常会被转发给调用者,如果这个consumer的accept抛出异常,after不会被执行 * @param after the operation to perform after this operation *

after是这个动作完成后,下一个做的动作 * @return a composed {@code Consumer} that performs in sequence this * operation followed by the {@code after} operation * @throws NullPointerException if {@code after} is null */ default Consumer andThen(Consumer after) { Objects.requireNonNull(after); return (T t) -> { accept(t); after.accept(t); }; }

测试

		Consumer consumer=new Consumer() {
			@Override
	        public void accept(String t) {
	                System.out.println(t+"123");
	        }
		};
		consumer.accept("abc");
		
		Consumer consumer2=new Consumer() {
			@Override
	        public void accept(String t) {
	                System.out.println(t+"456");
	        }
		};
		consumer2.accept("abc");
		
		Consumer consumer3=consumer.andThen(consumer2);
		consumer3.accept("abc");

 看一看到consumer3的效果=consumer1后面执行consumer2

abc123
abc456
abc123
abc456

经典应用-iterable

可以看到iterable接口的forEach方法使用了consumer类,传入一个consumer类型的参数action,

对iterable的所有元素作为参数执行下面的action.accept这个方法

    /**
     * Performs the given action for each element of the {@code Iterable}
     * until all elements have been processed or the action throws an
     * exception.  Unless otherwise specified by the implementing class,
     * actions are performed in the order of iteration (if an iteration order
     * is specified).  Exceptions thrown by the action are relayed to the
     * caller.
     * 

默认方法,对iterable的所有元素作为参数执行下面的action.accept这个方法,直到抛出异常 *

除非这个方法被覆盖的方法重写了,元素以iterator的顺序被执行 *

被抛出的异常会被转发给调用者 * @implSpec *

这个方法等价于下面的这个代码 *

{@code
     *     for (T t : this)
     *         action.accept(t);
     * }
* * @param action iterable里的所有元素,都作为参数,一个个给action对象,然后调用它的accept方法 * @throws NullPointerException if the specified action is null * @since 1.8 */ default void forEach(Consumer action) { Objects.requireNonNull(action); for (T t : this) { action.accept(t); } }

扩展类介绍

Consumer的accept只接受一个参数,那如果要是想使用多个参数要怎么办?jdk8又提供了一个BiConsumer接口类,该类与Consumer的区别是可以接受2个参数。

jdk8还对Consumer和BiConsumer各提供了3个常用的相关接口类,见下表:

类名 描述
IntConsumer 接受单个int型参数的Consumer操作
DoubleConsumer 接受单个double型参数的Consumer操作
LongConsumer 接受单个long型参数的Consumer操作
ObjIntConsumer 接受2个int型参数的Consumer操作,不支持andThen方法
ObjDoubleConsumer 接受2个double型参数的Consumer操作,不支持andThen方法
ObjLongConsumer 接受2个long型参数的Consumer操作,不支持andThen方法

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

你可能感兴趣的:(源码分析,java)