在前面一篇文章中简单介绍了Lambda表达式,我们知道使用Lambda表达式是需要使用函数式接口的,那么岂不是在开发过程中需要定义许多函数式接口?当时我就有这个疑问,每次还需要自定义一个纯函数式接口,也挺烦的。
后来了解到其实不然,Java8 其实已经为我们定义好了4类内置函数式接口,这4类接口已经可以解决我们开发过程中绝大部分的问题,只有一小部分比较特殊的情况需要我们自己去定义函数式接口,本文就简单来学习一下Java8内置的4大核心函数式接口。
一、Consumer<T>:消费型接口(void accept(T t))
@FunctionalInterface public interface Consumer<T> { /** * Performs this operation on the given argument. * @param t the input argument */ void accept(T t); }
// 消费型接口Consumer<T> @Test public void test1 () { consumo(500, (x) -> System.out.println(x)); } public void consumo (double money, Consumer<Double> c) { c.accept(money); }
这是一个简单的例子,我们需要了解的就是我们把 lambda 表达式 (x) -> System.out.println(x) 当做一个参数(参数为函数),传递给了 方法 consumo,方法 consumo 里是用 消费型接口 Consumer 接参的,然后就可以直接调用 消费型接口 的 accept 方法。
以上为消费型接口,有参数,无返回值类型的接口。
二、Supplier<T>:供给型接口(T get())
@FunctionalInterface public interface Supplier<T> { /** * Gets a result. * @return a result */ T get(); }
// 供给型接口,Supplier<T> @Test public void test2 () { Random ran = new Random(); List<Integer> list = supplier(10, () -> ran.nextInt(10)); for (Integer i : list) { System.out.println(i); } } /** * 随机产生sum个数量得集合 * @param sum 集合内元素个数 * @param sup * @return */ public List<Integer> supplier(int sum, Supplier<Integer> sup){ List<Integer> list = new ArrayList<Integer>(); for (int i = 0; i < sum; i++) { list.add(sup.get()); } return list; }
具体同上面分析咯,这就是一个供给类型的接口,只有产出,没有输入,就是只有返回值,没有入参。
三、Function<T, R>:函数型接口(R apply(T t))
@FunctionalInterface public interface Function<T, R> { /** * Applies this function to the given argument. * @param t the function argument * @return the function result */ R apply(T t); }
import java.util.function.Function; public class Test2 { // 函数型接口:Function<R, T> public static void main(String[] args) { String newStr = strHandler("abc",(str)->str.toUpperCase()); System.out.println(newStr); newStr = strHandler(" abc ",(str)->str.trim()); System.out.println(newStr); } /** * 字符串操作 * @param str 需要处理得字符串 * @param fun Function接口 * @return 处理之后得字符传 */ public static String strHandler(String str, Function<String,String> fun){ return fun.apply(str); } }
我们可以看到第一个输出:ABC,第二个输出:abc。这就是一个函数型接口,输入一个类型得参数,输出一个类型得参数,当然两种类型可以一致。
四、Predicate<T>:断言型接口(boolean test(T t))
@FunctionalInterface public interface Predicate<T> { /** * Evaluates this predicate on the given argument. * @param t the input argument * @return {@code true} if the input argument matches the predicate, * otherwise {@code false} */ boolean test(T t); }
import java.util.function.Predicate; public class Test2 { // 断言型接口:Predicate<T> public static void main(String[] args) { List<String> list = Arrays.asList("hello","jiangshuying","lambda","www","ok","q"); List<String> ret = filterStr(list,(str)->str.length()>2); System.out.println(ret); } /** * 过滤集合 * @param list * @param pre * @return */ public static List<String> filterStr(List<String> list, Predicate<String> pre){ ArrayList<String> arrayList = new ArrayList<>(); for(String str:list){ if(pre.test(str)) { arrayList.add(str); } } return arrayList; } }
上面例子判断一些字符串数组判断长度>2的字符串,这就是一个断言型接口,输入一个参数,输出一个boolean类型得返回值。
五、其他类型的一些函数式接口
除了上述得4种类型得接口外还有其他的一些接口供我们使用:
1、.BiFunction<T, U, R> :参数类型有2个,为T,U,返回值为R,其中方法为R apply(T t, U u)
2、.UnaryOperator<T>(Function子接口):参数为T,对参数为T的对象进行一元操作,并返回T类型结果,其中方法为T apply(T t)
3、.BinaryOperator<T>(BiFunction子接口):参数为T,对参数为T得对象进行二元操作,并返回T类型得结果,其中方法为T apply(T t1, T t2)
4、.BiConsumcr(T, U) :参数为T,U无返回值,其中方法为 void accept(T t, U u)
5、.ToIntFunction<T>、ToLongFunction<T>、ToDoubleFunction<T>:参数类型为T,返回值分别为int,long,double,分别计算int,long,double得函数。
6、.IntFunction<R>、LongFunction<R>、DoubleFunction<R>:参数分别为int,long,double,返回值为R。
以上就是Java8内置得核心函数式接口,其中包括了大部分得方法类型,所以可以在使用得时候根据不同得使用场景去选择不同得接口使用。