Java的Lambda表示式

scu醬油仔發表於2019-01-19

Java的Lambda表示式

1. 什麼是Lambda表示式

簡單的說,Lambda表示式就是匿名方法。Lambda表示式讓程式設計師能夠使用更加簡潔的程式碼,但是同樣也使程式碼的可讀性比較差。

Lambda表示式也叫做匿名方法或者閉包。

2. 和匿名內部類做對比

Lambda是匿名方法,這個時候我們會想想到匿名內部類,我們來回想一下匿名內部類的用法,比如下面的程式碼就是使用匿名內部類實現了一個執行緒。

public class Test {
  public static void main(String[] args) {
    Thread t = new Thread(new Runnable() {
      @Override
      public void run() {
        System.out.println("執行緒:" + Thread.currentThread().getName());
      }
    });
    t.start();
  }
}

我們一般的做法是寫一個Runnable介面的實現類,然後new一個實現類再傳給Thread的構造器。如下:

public class Test {

  public static void main(String[] args) {
    MyThread myThread = new MyThread();
    Thread t = new Thread(myThread);
    t.start();
  }

  static class MyThread implements Runnable {
    @Override
    public void run() {
      System.out.println("執行緒:" + Thread.currentThread().getName());
    }
  }
  
}

可以看到使用匿名內部類的話就省略了新建Runnable介面的實現類這一步驟。

3. 使用Lambda表示式

上面使用匿名內部類的寫法,如果使用Lambda表示式可以寫成下面這樣:

public class Test {
  public static void main(String[] args) {
    Thread t = new Thread(() ->  {
      System.out.println("執行緒:" + Thread.currentThread().getName());
    });
    t.start();
  }
}

這樣有一個問題,如果介面裡面有多個方法,那麼Lambda表示式怎麼知道實現的是哪個方法呢?我們通過程式碼測試一下:

package com.wangjun.othersOfJava;

public class LambdaTest {

    public static void main(String[] args) {
        Animal a = () -> {  // 編譯報錯:The target type of this expression must be a functional interface
            System.out.println("狗狗吃飯");
        };
        a.eat();
    }
    interface Animal {
        public void eat();
        public void duty();
    }
}

可以看到編譯報錯,這個提到一個functional interface,就是函式式介面。函式式介面就是隻有一個抽象方法的介面。這樣,就不難理解了,原來Lambda表示式只支援函式式介面。

4. Lambda表示式使用的幾種方式

package com.wangjun.othersOfJava;

public class LambdaTest {
    
    public static void main(String[] args) {
        
        // 帶型別
        Animal a1 = (String str) -> {
            System.out.println("狗狗吃飯:" + str);
        };
        // 不帶型別
        Animal a2 = (str) -> {
            System.out.println("狗狗吃飯:" + str);
        };
        // 不帶括號
        Animal a3 = str -> {
            System.out.println("狗狗吃飯:" + str);
        };
        // 不帶大括號
        Animal a4 = str -> System.out.println("狗狗吃飯:" + str);
        a1.eat("火腿腸");
        a2.eat("牛肉");
        a3.eat("麵條");
        a4.eat("米飯");
        
        // 使用return返回
        Person p1 = () -> {
            return "老師的職責:教書育人!";
        };
        // 直接返回
        Person p2 = () -> "醫生的職責:救死扶傷!";
        System.out.println(p1.duty());
        System.out.println(p2.duty());
    }
    
    // 沒有返回值
    interface Animal {
        public void eat(String str);
    }
    // 有返回值
    interface Person {
        public String duty();
    }
}

5. Java的雙冒號表示式

JDK8中有雙冒號的用法,就是把方法當做引數傳到stream內部,使stream的每個元素都傳入到該方法裡面執行一下。下面通過遍歷一個List來說明一下雙冒號和Lambda表示式使用方式的不同。

package com.wangjun.othersOfJava;

import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;

public class LambdaTest {
    
  public static void printStr(String str) {
    System.out.println(str);
  }

  public static void main(String[] args) {

    List<String> list = Arrays.asList("aaa","bbb","ccc");
    // 1.通常的遍歷方式
    for(String str: list) {
      LambdaTest.printStr(str);
    }
    // 2.使用Lambda表示式遍歷
    list.forEach(str -> {
      LambdaTest.printStr(str);
    });
    // 3.使用::遍歷
    list.forEach(LambdaTest::printStr);
    // 下面的方法和上面等價,使用的是函數語言程式設計
    Consumer<String> methodParam = LambdaTest::printStr; //方法引數
    list.forEach(x -> methodParam.accept(x));//方法執行accept
  }
}

相關文章