目录
简介
Consumer是一个函数式接口,它有一个需要覆盖的方法accept,代表接受一个参数输入且没有任何返回值的操作。不同于其它的函数式接口,Consumer期望通过方法的实现来执行具体的操作。
accept方法
可以看到它接受一个参数,什么都不返还
/** * Performs this operation on the given argument. * <p>接受一个参数,覆盖的方法对它做出动作,什么都不返回 * @param t the input argument */ void accept(T t);
具体使用例子
Consumer<String> consumer=new Consumer<String>() { @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. * <p>返回一个组合过的consumer实例, * <p>它的accept方法是先调用this这个consumer的accept方法,然后调用after的accept方法 * <p>被抛出的异常会被转发给调用者,如果这个consumer的accept抛出异常,after不会被执行 * @param after the operation to perform after this operation * <p>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<T> andThen(Consumer<? super T> after) { Objects.requireNonNull(after); return (T t) -> { accept(t); after.accept(t); }; }
测试
Consumer<String> consumer=new Consumer<String>() { @Override public void accept(String t) { System.out.println(t+"123"); } }; consumer.accept("abc"); Consumer<String> consumer2=new Consumer<String>() { @Override public void accept(String t) { System.out.println(t+"456"); } }; consumer2.accept("abc"); Consumer<String> consumer3=consumer.andThen(consumer2); consumer3.accept("abc");
看一看到consumer3的效果=consumer1后面执行consumer2
abc123abc456abc123abc456
经典应用-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. * <p>默认方法,对iterable的所有元素作为参数执行下面的action.accept这个方法,直到抛出异常 * <p>除非这个方法被覆盖的方法重写了,元素以iterator的顺序被执行 * <p>被抛出的异常会被转发给调用者 * @implSpec * <p>这个方法等价于下面的这个代码 * <pre>{@code * for (T t : this) * action.accept(t); * }</pre> * * @param action iterable里的所有元素,都作为参数,一个个给action对象,然后调用它的accept方法 * @throws NullPointerException if the specified action is null * @since 1.8 */ default void forEach(Consumer<? super T> 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方法 |
来源:https://www.cnblogs.com/ZhangZiYangDeBoKe/p/11590651.html