深入淺出理解Java Lambda表達式之四大核心函數式的用法與范例

1.四大核心函數式接口

上一篇文章中說到瞭Lambda表達式中的基本語法,以及我們如何自定義函數式接口。但是在寫代碼的過程中,大傢可能會發現一個問題:當我們有一個新的需求時,可以去自定義一個函數式接口,然後再創建一個它的實現類定義一些相關的業務邏輯行為。那麼如果說我們有很多需求、這些需求可能還會不斷地變化,那麼我們豈不是每次都要去創建新的實現類、同時再去修改之前創建好的實現類中的業務代碼?這可太麻煩瞭吧。。。

所以呢,Java8就為我們提供瞭四大核心函數式接口,使用起來非常的方便。

1.1 Consumer<T> : 消費型接口

package com.szh.java8;
 
import org.junit.Test;
 
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
 
/*
 * 
 */
public class MyTest3 {
 
    //Consumer<T> : 消費型接口
    @Test
    public void test1() {
        happy(6666.66,(m) -> System.out.println("本地雙11共消費 " + m + " 元!!!"));
    }
 
    public void happy(double money, Consumer<Double> consumer) {
        consumer.accept(money);
    }
 
}

1.2 Supplier<T> : 供給型接口

package com.szh.java8;
 
import org.junit.Test;
 
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
 
/*
 *
 */
public class MyTest3 {
 
    //Supplier<T> : 供給型接口
    @Test
    public void test2() {
        List<Integer> numList = getNumList(10, () -> (int)(Math.random() * 100));
        for (Integer num : numList) {
            System.out.println(num);
        }
    }
 
    public List<Integer> getNumList(int num, Supplier<Integer> supplier) {
        List<Integer> list = new ArrayList<>();
 
        for (int i = 0; i < num; i++) {
            Integer n = supplier.get();
            list.add(n);
        }
 
        return list;
    }
 
}

1.3 Function<T, R> : 函數型接口

package com.szh.java8;
 
import org.junit.Test;
 
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
 
/*
 * 
 */
public class MyTest3 {
 
    //Function<T, R> : 函數型接口
    @Test
    public void test3() {
        String trimStr = strHandler("\t\t\t  張起靈-小哥   ", (str) -> str.trim());
        System.out.println(trimStr);
 
        String newStr = strHandler("我喜歡看盜墓筆記呀!!!",(str) -> str.substring(4,8));
        System.out.println(newStr);
    }
 
    public String strHandler(String str, Function<String,String> function) {
        return function.apply(str);
    }
 
}

1.4 Predicate<T> : 斷言型接口

package com.szh.java8;
 
import org.junit.Test;
 
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
 
/*
 *
 */
public class MyTest3 {
 
    //Predicate<T> : 斷言型接口
    @Test
    public void test4() {
        List<String> list = Arrays.asList("Hello","張起靈-小哥","HashMap","jdk8","List","Set");
        List<String> stringList = filterStr(list, (s) -> s.length() > 5);
 
        for (String string : stringList) {
            System.out.println(string);
        }
    }
 
    public List<String> filterStr(List<String> strings, Predicate<String> predicate) {
        List<String> strList = new ArrayList<>();
 
        for (String str : strings) {
            if (predicate.test(str)) {
                strList.add(str);
            }
        }
 
        return strList;
    }
}

除此之外,還有一些其他的函數式接口,它們有一部分是上面提到的四大核心函數式接口的子接口。

2.方法引用

當要傳遞給Lambda體的操作,已經有實現的方法瞭,可以使用方法引用! 方法引用:使用操作符 “ :: ” 將方法名和對象或類的名字分隔開來。

如下三種主要使用情況 :

  • 對象 :: 實例方法
  • 類 :: 靜態方法
  • 類 :: 實例方法

可以將方法引用理解為 Lambda 表達式的另外一種表現形式,方法引用所引用的方法的參數列表與返回值類型,需要與函數式接口中抽象方法的參數列表和返回值類型保持一致!

2.1 對象 :: 實例方法

    @Test
    public void test1() {
        Consumer<String> con1 = (str) -> System.out.println(str);
        con1.accept("Hello World!!!");
 
        PrintStream ps = System.out;
        Consumer<String> con2 = ps::println;
        con2.accept("Hello Java8!!!");
 
        Consumer<String> con3 = System.out::println;
        con3.accept("Hello Lambda!!!");
    }

    @Test
    public void test2() {
        Employee emp = new Employee();
        emp.setName("張起靈");
        emp.setAge(18);
 
        Supplier<? extends Object> sup1 = () -> emp.getName();
        String str = (String) sup1.get();
        System.out.println(str);
 
        Supplier<Integer> sup2 = emp::getAge;
        Integer age = sup2.get();
        System.out.println(age);
    }
package com.szh.java8;
 
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
 
/**
 *
 */
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Employee {
 
    private Integer id;
    private String name;
    private Integer age;
    private Double salary;
 
    public Employee(Integer id) {
        this.id = id;
    }
 
    public Employee(Integer id,String name) {
        this.id = id;
        this.name = name;
    }
 
}

2.2 類 :: 靜態方法

    @Test
    public void test3() {
        Comparator<Integer> com1 = (x,y) -> Integer.compare(x,y);
        System.out.println(com1.compare(10, 20));
 
        Comparator<Integer> com2 = Integer::compare;
        System.out.println(com2.compare(300, 110));
    }

2.3 類 :: 實例方法

若 Lambda 的參數列表的第一個參數是實例方法的調用者,第二個參數(或無參)是實例方法的參數時,格式: ClassName::MethodName

    @Test
    public void test4() {
        BiPredicate<String,String> bp1 = (str1,str2) -> str1.equals(str2);
        System.out.println(bp1.test("Hello", "hello"));
 
        BiPredicate<String,String> bp2 = String::equals;
        System.out.println(bp2.test("Java", "Java"));
    }

3.構造器引用

格式 : ClassName::new

與函數式接口相結合,自動與函數式接口中方法兼容。構造器的參數列表,需要與函數式接口中參數列表保持一致!

代碼中Employee類參考上面的案例。

    @Test
    public void test5() {
        //無參構造器
        Supplier<Employee> sup1 = () -> new Employee();
        System.out.println(sup1.get());
 
        //無參構造器
        Supplier<Employee> sup2 = Employee::new;
        System.out.println(sup2.get());
 
        //一個參數構造器
        Function<Integer,Employee> function = Employee::new;
        Employee employee = function.apply(1001);
        System.out.println(employee);
 
        //兩個參數構造器
        BiFunction<Integer,String,Employee> biFunction = Employee::new;
        Employee emp = biFunction.apply(1001, "張起靈");
        System.out.println(emp);
    }

4.數組引用

格式 :類型[] :: new

    @Test
    public void test6() {
        Function<Integer,String[]> fun = (x) -> new String[x];
        String[] strings = fun.apply(10);
        System.out.println(strings.length);
 
        Function<Integer,String[]> fun2 = String[]::new;
        String[] strArray = fun2.apply(50);
        System.out.println(strArray.length);
    }

以上就是深入淺出理解Java Lambda表達式之四大核心函數式的用法與范例的詳細內容,更多關於Java Lambda表達式的資料請關註WalkonNet其它相關文章!

推薦閱讀: