1. 程式人生 > >jdk8-lambda表達式的使用

jdk8-lambda表達式的使用

更改 個數 局部變量 反編譯 each read int function 聚集函數

1, 遍歷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);
  });

2, 利用函數式接口實現匿名內部類

實現一個多線程

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

還可以

(params) -> expression
(params) -> statement
(params) -> { statements }

3, Predicate接口

Predicate是jdk8 中的新增接口, 共有5個方法,

//Returns a predicate which evaluates to true only if this predicate
//and the provided predicate both evaluate to true.
and(Predicate<? super T> p) 

//Returns a predicate which negates the result of this predicate.
negate() //Returns a predicate which evaluates to true if either //this predicate or the provided predicate evaluates to true or(Predicate<? super T> p) //Returns true if the input object matches some criteria test(T t) //Returns a predicate that evaluates to true if both or neither //of the component predicates evaluate to true
xor(Predicate<? super T> p)

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

    @Test
    public void test15() {
        List<String> languages = Arrays.asList("Java", "Scala", "C++", "Haskell", "Lisp");

        System.out.println("Languages which starts with J :");
        filter(languages, (str) -> ((String) str).startsWith("J"));

        System.out.println("Languages 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);

    }

    public static void filter(List<String> names, Predicate condition) {
        for (String name : names) {
            if (condition.test(name)) {
                System.out.println(name + " ");
            }
        }
    }

4,Predicate接口中的 and, or, xor的使用

為邏輯判斷

    @Test
    public void test16() {
        Predicate<String> startWithJ = (n) -> n.startsWith("J");
        Predicate<String> fourLength = (n) -> n.length() == 4;
        
        List<String> languages = Arrays.asList("Java", "Scala", "C++", "Haskell", "Lisp");
        languages.stream().filter(startWithJ.and(fourLength))
                    .forEach(System.out::println);
    }

5, map, 允許將對象進行轉換, 比如, 可以更改list中的每個元素的值

    @Test
    public void test14() {
        List<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);

        // 可改變對象
        list.stream().map((i) -> i * 3).forEach(System.out::println);

        // 不可改變元有對象
        list.forEach(i -> i = i * 3);
        list.forEach(System.out::println);
        ;
    }

只在本次調用中有效, 並不會改變原有的list

6, reduce, 用來將值進行合並, 又稱折疊操作, Map和Reduce操作是函數式編程的核心操作

SQL中類似 sum()、avg() 或者 count() 的聚集函數,實際上就是 reduce 操作,因為它們接收多個值並返回一個值。流API定義的 reduceh() 函數可以接受lambda表達式,並對所有值進行合並。IntStream這樣的類有類似 average()、count()、sum() 的內建方法來做 reduce 操作,也有mapToLong()、mapToDouble() 方法來做轉換

    @Test
    public void test17() {
        List<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);

        Integer integer = list.stream().map((i) -> i = i * 3)
            .reduce((sum, count) -> sum += count).get();
        
        System.out.println(integer);
    }

7, 通過過濾創建一個string, list

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

@Test
public void test2() {
  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);
}

過濾後將獲取一個新的列表

8, 對列表的每個元素使用 函數

    @Test
    public void test3() {
        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);
    }

9, 使用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);

10, 計算最值和平均值

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());

11, 使用方法引用, 不對參數做任何修改

Person

package com.lambda.usebean;

/**
 * 實體類Person
 * @author MingChenchen
 *
 */
public class Person {
    private String name;      //姓名
    private String location;  //地址

    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getLocation() {
        return location;
    }
    public void setLocation(String location) {
        this.location = location;
    }

    @Override
    public String toString() {
        // TODO Auto-generated method stub
        return "Person:" + name + "," + location;
    }
}

引用

//使用String默認的排序規則,比較的是Person的name字段
Comparator<Person> byName = Comparator.comparing(p -> p.getName());
//不用寫傳入參數,傳入的用Person來聲明
Comparator<Person> byName2 = Comparator.comparing(Person::getName);

關於lambda的註意:

1)lambda表達式僅能放入如下代碼:預定義使用了 @Functional 註釋的函數式接口,自帶一個抽象函數的方法,或者SAM(Single Abstract Method 單個抽象方法)類型。這些稱為lambda表達式的目標類型,可以用作返回類型,或lambda目標代碼的參數。例如,若一個方法接收Runnable、Comparable或者 Callable 接口,都有單個抽象方法,可以傳入lambda表達式。類似的,如果一個方法接受聲明於 java.util.function 包內的接口,例如 Predicate、Function、Consumer 或 Supplier,那麽可以向其傳lambda表達式。

2)lambda表達式內可以使用方法引用,僅當該方法不修改lambda表達式提供的參數。本例中的lambda表達式可以換為方法引用,因為這僅是一個參數相同的簡單方法調用。

list.forEach(n -> System.out.println(n));
list.forEach(System.out::println);  // 使用方法引用

然而,若對參數有任何修改,則不能使用方法引用,而需鍵入完整地lambda表達式,如下所示:

list.forEach((String s) -> System.out.println("*" + s + "*"));

事實上,可以省略這裏的lambda參數的類型聲明,編譯器可以從列表的類屬性推測出來。

3)lambda內部可以使用靜態、非靜態和局部變量,這稱為lambda內的變量捕獲。

4)Lambda表達式在Java中又稱為閉包或匿名函數,所以如果有同事把它叫閉包的時候,不用驚訝。

5)Lambda方法在編譯器內部被翻譯成私有方法,並派發 invokedynamic 字節碼指令來進行調用。可以使用JDK中的 javap 工具來反編譯class文件。使用 javap -p 或 javap -c -v 命令來看一看lambda表達式生成的字節碼。大致應該長這樣:

private static java.lang.Object lambda$0(java.lang.String);

6)lambda表達式有個限制,那就是只能引用 final 或 final 局部變量,這就是說不能在lambda內部修改定義在域外的變量。

Compile time error : "local variables referenced from a lambda expression must be final or effectively final"

另外,只是訪問它而不作修改是可以的,如下所示:

List<Integer> primes = Arrays.asList(new Integer[]{2, 3,5,7});
int factor = 2;
primes.forEach(element -> { System.out.println(factor*element); });

因此,它看起來更像不可變閉包,類似於Python。

參考:

http://www.importnew.com/16436.html

http://ifeve.com/stream/

http://ifeve.com/lambda/

http://blog.csdn.net/jinzhencs/article/details/50748202

http://ifeve.com/predicate-and-consumer-interface-in-java-util-function-package-in-java-8/

http://www.jdon.com/idea/java/10-example-of-lambda-expressions-in-java8.html

jdk8-lambda表達式的使用