Java8新特性代码示例(附注释)- 方法引用,Optional, Stream

Posted 那个人样子好怪

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java8新特性代码示例(附注释)- 方法引用,Optional, Stream相关的知识,希望对你有一定的参考价值。

  /**
     * java8中的函数式接口,java中规定:函数式接口必须只有一个抽象方法,可以有多个非抽象方法,同时,如果继承实现了
     * Object中的方法,那么也是合法的
     * <p>
     * 函数式接口可以使用lambda来创建
     */
    @FunctionalInterface
    interface MyService {
        void sayMessage(String msg);

        @Override
        boolean equals(Object obj);
    }

    /**
     * 入口
     * @param args
     */
    public static void main(String[] args) {
        printSeperator();
        testMehtodRef();
        printSeperator();

        testSupplierConsumer();
        printSeperator();

        testOptional();
        printSeperator();

        testStream();
        printSeperator();
    }

    /**
     * 测试Optional类
     */
    private static void testOptional() {
        Optional<String> optional1 = Optional.ofNullable(null);
        //如果optional确实存在值,那么该函数返回值为true
        if (optional1.isPresent()) {
            System.out.println("optional1 present!");
        }

        //如果optional没有值,那么默认返回传入的other
        String res = optional1.orElse("default");
        System.out.println("res's value after orElse:" + res);
        try {
            /**
             * 如果没有该值,那么抛出{@link NoSuchElementException}
             */
            res = optional1.orElseThrow();
            System.out.println("res's value after orElseThrow:" + res);
        } catch (NoSuchElementException e) {
            System.out.println("orElseThrow occured!");
        }

        optional1 = Optional.ofNullable("i have a value");
        res = optional1.orElse("default");
        System.out.println("res's value after orElse:" + res);
    }

    /**
     * 测试{@link java.util.function.Supplier} and {@link java.util.function.Consumer}
     */
    private static void testSupplierConsumer() {
        Supplier<String> supplier1 = new Supplier<String>() {
            @Override
            public String get() {
                return new String("hello world from supplier1");
            }
        };

        //lambda式的supplier,supplier的规定是无参数,且返回一个对象,因此满足该条件
        Supplier<String> supplier2 = () -> new String("hello world from supplier2");

        //本质上supplier是一个容器,对一个方法的封装
        var str1 = getSupplier(supplier1);
        var str2 = getSupplier(supplier2);
        //方法引用默认返回一个supplier,因为该new不接受任何参数,返回一个对象,因此满足条件
        var str3 = getSupplier(String::new);
        System.out.println("str1: " + str1 + "   str2:" + str2 + "   str3:" + str3);

        /////////////////////////////////////////////////////////////////////////////////////////////
        Consumer<String> consumer1 = new Consumer<String>() {
            @Override
            public void accept(String s) {
                System.out.print(s + "|");
            }
        };
        //lambda式的Consumer,Consumer的约定是接受一个参数,返回值为void,因此满足条件
        Consumer<String> consumer2 = (s) -> System.out.print(s + "^");
        var list = getStrList();
        list.forEach(consumer1);
        System.out.println();
        list.forEach(consumer2);
        System.out.println();

        /////////////////////////////////////////////////////////////////////////////////////////////
        //Predicate谓词接口,用于判断是否符合某个条件,约定是:接受一个参数,返回值为boolean
        Predicate<Integer> predicate1 = new Predicate<Integer>() {
            @Override
            public boolean test(Integer integer) {
                return integer > 2;
            }
        };

        //lambda式的谓词接口
        Predicate<Integer> predicate2 = (num) -> num > 2;
        var numList = getIntList();
        var res = numList.stream().filter(predicate1).collect(Collectors.toList());
        System.out.println(res);
        res = numList.stream().filter(predicate2).collect(Collectors.toList());
        System.out.println(res);

        /////////////////////////////////////////////////////////////////////////////////////////////
        //Function接口:这是一个功能性的接口,主要用于把一种类型的数据处理完成后,可能得到的结果是另外一种类型
        //本质上是上面那些接口的超集,第一个参数为接口的类型,第二个参数为转换后的类型
        Function<String, Character> func1 = new Function<String, Character>() {
            @Override
            public Character apply(String s) {
                if (s.length() <= 2) {
                    return 'a';
                }
                return s.charAt(2);
            }
        };

        Function<String, Character> func2 = (s) -> {
            return s.length() <= 2 ? 'a' : s.charAt(2);
        };
        var varList = getStrList();
        var res2 = varList.stream().map(func1).collect(Collectors.toList());
        System.out.println(res2);
        res2 = varList.stream().map(func2).collect(Collectors.toList());
        System.out.println(res2);
    }

    /**
     * 测试{@link java.util.stream.Stream}里面的函数接口,
     * Stream里面的函数接口非常的多,类似于SQL,分为两种:
     * 1 中间函数 immediate function
     * 2 最终函数(规约) terminal function
     */
    private static void testStream() {
        //将一个数字list,先过滤到大于3的数,然后转换为对应的字母,最后在大写,最后重新收集为一个List
        List<Integer> numList = getIntList();
        List<String> res = numList.stream().filter((num) -> num > 3).map(num -> String.valueOf((char)('a' + num)))
                .map(str -> String.valueOf((char)('A' + str.charAt(0) - 'a'))).collect(Collectors.toList());
        System.out.println(res);

        final int size = 100;
        int[] arr = new int[size];
        Random random = new Random();
        for(int i = 0;i < size;++i){
            arr[i] = random.nextInt(1000);
        }
        //对numArr进行排序,然后得到最前面的5个数据,最后得到最大值
        var res2 = Arrays.stream(arr).sorted().limit(5).min();
        System.out.println("res2:"+ res2.orElseThrow());
    }

    /**
     * 辅助类
     *
     * @param supplier
     * @param <T>
     * @return
     */
    private static <T> T getSupplier(Supplier<T> supplier) {
        return supplier.get();
    }

    /**
     * 测试方法引用的用法
     */
    private static void testMehtodRef() {
        List<String> strs = new ArrayList<>();
        strs.add("one");
        strs.add("two");
        strs.add("three");
        strs.add("four");

        //printSomething本质上是一个Consumer
        strs.forEach(HelloJava8::printSomething);
        System.out.println();
    }


    /**
     * 打印一些东西
     *
     * @param s
     */
    private static void printSomething(String s) {
        System.out.print(s + "	");
    }


    private static void printSeperator() {
        System.out.println(SEPERATOR);
    }

    /**
     * 辅助方法,获辅助list
     *
     * @return
     */
    private static List<String> getStrList() {
        List<String> strs = new ArrayList<>();
        strs.add("one");
        strs.add("two");
        strs.add("three");
        strs.add("four");
        return strs;
    }

    /**
     * 获取辅助list
     *
     * @return
     */
    private static List<Integer> getIntList() {
        List<Integer> intList = new ArrayList<>();
        for (int i = 1; i <= 10; ++i) {
            intList.add(i);
        }
        return intList;
    }

以上是关于Java8新特性代码示例(附注释)- 方法引用,Optional, Stream的主要内容,如果未能解决你的问题,请参考以下文章

Android 使用Java8新特性之"方法引用"

JAVA8 十大新特性详解

Java8新特性:函数式接口,方法与构造器引用

JAVA8 十大新特性

JAVA8新特性——方法引用

Java8 新特性 -- Lambda表达式:函数式接口方法的默认实现和静态方法方法引用注解类型推测Optional类Stream类调用JavaScriptBase64