1. 程式人生 > >深入理解Java8 Lambda表示式

深入理解Java8 Lambda表示式

一、Lambda表示式是什麼?

Lambda表示式有兩個特點:一是匿名函式;二是可傳遞。

匿名函式的應用場景是:

通常在需要一個函式,但又不想費神去命名一個函式的場合下使用。Lambda表示式所表示的匿名函式的內容應該是很簡單的,如果複雜的話,乾脆就去重新定義一個函數了。

可傳遞的應用場景是:

就是將Lambda表示式作為引數傳遞給其他函式,Lambda表示式作為一種更緊湊的程式碼風格,使Java的語言表達能力能到提升。

二、Lambda表示式語法

Lambda表示式在Java語言中引入了一個新的語法元素和操作符。這個操作符是“->”,該操作符被稱為Lambda操作符或箭頭操作符,它將Lambda分為兩個部分:

左側:指定了Lambda表示式所需要的所有引數

右側:制定了Lambda體,即Lambda表示式所要執行的功能。

三、Lambda表示式常見的語法格式

3.1、無參、無返回值,Lambda體只需要一條語句。

Runnable r = () -> System.out.println("Hello Lambda!");

3.2、Lambda需要一個引數

Consumer<String> con = (x) -> System.out.println(x);

3.3、Lambda只需要一個引數時,引數的小括號可以省略

Consumer<String> con = x -> System.out.println(x);

3.4、Lambda需要兩個引數,並且有返回值

Comparator<Integer> com = (x, y) -> {

System.out.println("函式式介面");

return Integer.compare(x, y);

};

3.5、當Lambda體只有一條語句時,return與大括號可以省略

Comparator<Integer> com = (x, y) -> Integer.compare(x, y);

3.6、資料型別可以省略,因為可由編譯器推斷得出,稱為型別推斷

BinaryOperator<Long> operator = (Long x, Long y) -> {

System.out.println("實現函式介面方法");

return x + y;

};

四、Lambda表示式實戰

4.1、實現一個多執行緒

new Thread(() -> System.out.println("In Java8!")).start();

4.2、遍歷list集合

List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
// 直接列印
list.forEach(System.out::println);

// 取值分別操作
list.forEach(i -> {
    System.out.println(i * 3);
});

4.3、map函式,允許將物件進行轉換。比如,可以更改list中的每個元素的值

List< Integer > list = Arrays.asList(1, 2, 3);
// 可改變物件
list.stream().map((i) -> i * 3).forEach(System.out::println);

// 不可改變原有物件
list.forEach(i -> i = i * 3);
list.forEach(System.out::println);

4.4、reduce函式,用來將值進行合併,map和reduce函式式函式式變成的核心。

List< Integer > list = Arrays.asList(1, 2, 3);
Integer integer = list.stream().map((i) -> i = i * 3).reduce((sum, count) -> sum += count).get();
System.out.println(integer);

reduce的更多用法

// 字串連線,concat = "ABCD"
String concat = Stream.of("A", "B", "C", "D").reduce("", String::concat);

// 求最小值,minValue = -3.0
double minValue = Stream.of(-1.5, 1.0, -3.0, -2.0).reduce(Double.MAX_VALUE, Double::min);

// 求和,sumValue = 10, 有起始值
int sumValue = Stream.of(1, 2, 3, 4).reduce(0, Integer::sum);

// 求和,sumValue = 10, 無起始值
sumValue = Stream.of(1, 2, 3, 4).reduce(Integer::sum).get();

// 過濾,字串連線,concat = "ace"
concat = Stream.of("a", "B", "c", "D", "e", "F").filter(x -> x.compareTo("Z") > 0).reduce("", String:concat);

4.5、過濾

過濾是Java開發者在大規模集合上的一個常用操作,而現在使用lambda表示式和流API過濾大規模資料集合是驚人的簡單。流提供了一個 filter() 方法,接受一個 Predicate 物件,即可以傳入一個lambda表示式作為過濾邏輯。下面的例子是用lambda表示式過濾Java集合,將幫助理解。

List<String> strList = Arrays.asList("abc", "eqwr", "bcd", "qb" , "ehdc", "jk");
List<String> filtered = strList.stream().filter(x -> x.length()> 2).collect(Collectors.toList());
System.out.printf("Original List : %s, filtered list : %s %n", strList, filtered);

4.6、Predicate介面

Predicate是jdk8中的新增介面,共有5個方法。

and(Predicate<? super T> p)  negate()  or(Predicate<? super T> p)  test(T t)  xor(Predicate<? super T> p)

該介面除了test方法是抽象方法, 其餘都是default方法, 該介面可接受一個 lambda表示式, 其實就是實現了test介面的一個匿名類

public static void main(String[] args) {
    List<String> languages = Arrays.asList("Java", "Scala", "C++", "Haskell", "Lisp");
    System.out.println("Language which starts with J :");
    filter(languages, (str) -> ((String) str).startsWith("J"));

    System.out.println("Language which ends with a :");
    filter(languages, (str) -> ((String) str).endsWith("a"));

    System.out.println("Print all languages :");
    filter(languages, (str) -> true);

    System.out.println("Print no language :");
    filter(languages, (str) -> false);

    System.out.println("Print language whose length greater than 4:");
    filter(languages, (str) -> ((String) str).length() > 4);


    Predicate<String> startWithJ = (n) -> n.startsWith("J");
    Predicate<String> fourLength = (n) -> n.length() == 4;
    languages.stream().filter(startWithJ.and(fourLength)).forEach(System.out::println);
}

public static void filter(List<String> names, Predicate condition) {
    names.stream().filter(x -> condition.test(x)).forEach(x -> System.out.println(x + ""));
}

4.7、對列表的每個元素使用函式

List<String> strList = Arrays.asList("abc", "eqwr", "bcd", "qb" , "ehdc", "jk");
String collect = strList.stream().map(x -> x.toUpperCase()).collect(Collectors.joining(", "));
System.out.printf("filtered list : %s %n",  collect);

4.8、使用distinct進行去重

List<Integer> numbers = Arrays.asList(9, 10, 3, 4, 7, 3, 4);
List<Integer> distinct = numbers.stream().map( i -> i*i).distinct().collect(Collectors.toList());
System.out.printf("Original List : %s,  Square Without duplicates : %s %n", numbers, distinct);

4.9、計算最值和平均值

IntStream、LongStream 和 DoubleStream 等流的類中,有個非常有用的方法叫做 summaryStatistics() 。可以返回 IntSummaryStatistics、LongSummaryStatistics 或者 DoubleSummaryStatistic s,描述流中元素的各種摘要資料。在本例中,我們用這個方法來計算列表的最大值和最小值。它也有 getSum() 和 getAverage() 方法來獲得列表的所有元素的總和及平均值。

//獲取數字的個數、最小值、最大值、總和以及平均值
List<Integer> primes = Arrays.asList(2, 3, 5, 7, 11, 13, 17, 19, 23, 29);
IntSummaryStatistics stats = primes.stream().mapToInt((x) -> x).summaryStatistics();
System.out.println("Highest prime number in List : " + stats.getMax());
System.out.println("Lowest prime number in List : " + stats.getMin());
System.out.println("Sum of all prime numbers : " + stats.getSum());
System.out.println("Average of all prime numbers : " + stats.getAverage());