Java8 lambda表達式10個示例


 1. 實現Runnable線程案例

使用() -> {} 替代匿名類:

//Before Java 8:
new Thread(new Runnable() {
    @Override
    public void run() {
        System.out.println("Before Java8 ");
    }
}).start();

//Java 8 way:
new Thread( () -> System.out.println("In Java8!") ).start();
     
Output:
too much code, for too little to do
Lambda expression rocks !!

你可以使用 下面語法實現Lambda:

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

如果你的方法並不改變任何方法參數,比如只是輸出,那么可以簡寫如下:

() -> System.out.println("Hello Lambda Expressions");

如果你的方法接受兩個方法參數,如下:

(int even, int odd) -> even + odd

2.實現事件處理

如果你曾經做過Swing 編程,你將永遠不會忘記編寫事件偵聽器代碼。使用lambda表達式如下所示寫出更好的事件偵聽器的代碼。

// Before Java 8:
JButton show =  new JButton("Show");
show.addActionListener(new ActionListener() {
     @Override
     public void actionPerformed(ActionEvent e) {
           System.out.println("without lambda expression is boring");
        }
     });


// Java 8 way:
show.addActionListener((e) -> {
    System.out.println("Action !! Lambda expressions Rocks");
});

在java 8中你可以使用Lambda表達式替代丑陋的匿名類。

 

3.使用Lambda表達式遍歷List集合

//Prior Java 8 :
List features = Arrays.asList("Lambdas", "Default Method", 
"Stream API", "Date and Time API"); for (String feature : features) { System.out.println(feature); } //In Java 8: List features = Arrays.asList("Lambdas", "Default Method", "Stream API",
"Date and Time API"); features.forEach(n -> System.out.println(n)); // Even better use Method reference feature of Java 8 // method reference is denoted by :: (double colon) operator // looks similar to score resolution operator of C++ features.forEach(System.out::println); Output: Lambdas Default Method Stream API Date and Time API

方法引用是使用兩個冒號::這個操作符號。

4.使用Lambda表達式和函數接口

為了支持函數編程,Java 8加入了一個新的包java.util.function,其中有一個接口java.util.function.Predicate是支持Lambda函數編程:

public static void main(args[]){
  List languages = Arrays.asList("Java", "Scala", "C++", "Haskell", "Lisp");

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

  System.out.println("Languages which ends with a ");
  filter(languages, (str)->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)->str.length() > 4);
}

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

Output:
Languages which starts with J :
Java
Languages which ends with a
Java
Scala
Print all languages :
Java
Scala
C++
Haskell
Lisp
Print no language :
Print language whose length greater than 4:
Scala
Haskell

//Even better
 public static void filter(List names, Predicate condition) {
    names.stream().filter((name) -> (condition.test(name)))
.forEach((name) -> {System.out.println(name + " "); }); }

你能看到來自Stream API 的filter方法能夠接受 Predicate參數, 能夠允許測試多個條件。

 

5.復雜的結合Predicate 使用

java.util.function.Predicate提供and(), or() 和 xor()可以進行邏輯操作,比如為了得到一串字符串中以"J"開頭的4個長度:

// We can even combine Predicate using and(), or() And xor() logical functions
 // for example to find names, which starts with J and four letters long, you
 // can pass combination of two Predicate
 Predicate<String> startsWithJ = (n) -> n.startsWith("J");
 Predicate<String> fourLetterLong = (n) -> n.length() == 4;
   
 names.stream()
      .filter(startsWithJ.and(fourLetterLong))
      .forEach((n) -> System.out.print("\nName, which starts with
'J' and four letter long is : "
+ n));

其中startsWithJ.and(fourLetterLong)是使用了AND邏輯操作。

 

6.使用Lambda實現Map 和 Reduce

最流行的函數編程概念是map,它允許你改變你的對象,在這個案例中,我們將costBeforeTeax集合中每個元素改變了增加一定的數值,我們將Lambda表達式 x -> x*x傳送map()方法,這將應用到stream中所有元素。然后我們使用 forEach() 打印出這個集合的元素.

// applying 12% VAT on each purchase
// Without lambda expressions:
List costBeforeTax = Arrays.asList(100, 200, 300, 400, 500);
for (Integer cost : costBeforeTax) {
      double price = cost + .12*cost;
      System.out.println(price);
}

// With Lambda expression:
List costBeforeTax = Arrays.asList(100, 200, 300, 400, 500);
costBeforeTax.stream().map((cost) -> cost + .12*cost)
.forEach(System.out::println); Output 112.0 224.0 336.0 448.0 560.0 112.0 224.0 336.0 448.0 560.0

reduce() 是將集合中所有值結合進一個,Reduce類似SQL語句中的sum(), avg() 或count(), 

// Applying 12% VAT on each purchase// Old way:List costBeforeTax =Arrays.asList(100, 200, 300, 400, 500);
double total =0;
for (Integer cost :costBeforeTax) {
 double price = cost + .12*cost;
 total = total + price;
 
}
System.out.println("Total : " + total);

// New way:List costBeforeTax =Arrays.asList(100, 200, 300, 400, 500);
double bill = costBeforeTax.stream().map((cost) -> cost + .12*cost)
.reduce((sum, cost) -> sum + cost)
.get();
System.out.println("Total : " + bill); OutputTotal :1680.0Total:1680.0

 

7.通過filtering 創建一個字符串String的集合

Filtering是對大型Collection操作的一個通用操作,Stream提供filter()方法,接受一個Predicate對象,意味着你能傳送lambda表達式作為一個過濾邏輯進入這個方法:

// Create a List with String more than 2 characters
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); Output : Original List : [abc, , bcd, , defg, jk], filtered list : [abc, bcd, defg]

 

8.對集合中每個元素應用函數

我們經常需要對集合中元素運用一定的功能,如表中的每個元素乘以或除以一個值等等.

// Convert String to Uppercase and join them using coma
List<String> G7 = Arrays.asList("USA", "Japan", "France", "Germany", 
                                "Italy", "U.K.","Canada");
String G7Countries = G7.stream().map(x -> x.toUpperCase())
.collect(Collectors.joining(", ")); System.out.println(G7Countries); Output : USA, JAPAN, FRANCE, GERMANY, ITALY, U.K., CANADA

上面是將字符串轉換為大寫,然后使用逗號串起來。

 

9.通過復制不同的值創建一個子列表

使用Stream的distinct()方法過濾集合中重復元素。

// Create List of square of all distinct numbers
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); Output : Original List : [9, 10, 3, 4, 7, 3, 4], Square Without
duplicates : [81, 100, 9, 16, 49]

 

10.計算List中的元素的最大值,最小值,總和及平均值

//Get count, min, max, sum, and average for numbers
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()); Output : Highest prime number in List : 29 Lowest prime number in List : 2 Sum of all prime numbers : 129 Average of all prime numbers : 12.9

Java 8 lambda表達式示例

我個人對Java 8發布非常激動,尤其是lambda表達式和流API。越來越多的了解它們,我能寫出更干凈的代碼。雖然一開始並不是這樣。第一次看到用lambda表達式寫出來的Java代碼時,我對這種神秘的語法感到非常失望,認為它們把Java搞得不可讀,但我錯了。花了一天時間做了一些lambda表達式和流API示例的練習后,我開心的看到了更清晰的Java代碼。這有點像學習泛型,第一次見的時候我很討厭它。我甚至繼續使用老版Java 1.4來處理集合,直到有一天,朋友跟我介紹了使用泛型的好處(才意識到它的好處)。所以基本立場就是,不要畏懼lambda表達式以及方法引用的神秘語法,做幾次練習,從集合類中提取、過濾數據之后,你就會喜歡上它。下面讓我們開啟學習Java 8 lambda表達式的學習之旅吧,首先從簡單例子開始。

例1、用lambda表達式實現Runnable

我開始使用Java 8時,首先做的就是使用lambda表達式替換匿名類,而實現Runnable接口是匿名類的最好示例。看一下Java 8之前的runnable實現方法,需要4行代碼,而使用lambda表達式只需要一行代碼。我們在這里做了什么呢?那就是用() -> {}代碼塊替代了整個匿名類

1
2
3
4
5
6
7
// Java 8之前:
new Thread( new Runnable() {
     @Override
     public void run() {
     System.out.println( "Before Java8, too much code for too little to do" );
     }
}).start();
1
2
//Java 8方式:
new Thread( () -> System.out.println( "In Java8, Lambda expression rocks !!" ) ).start();

輸出:

1
2
too much code, for too little to do
Lambda expression rocks !!

這個例子向我們展示了Java 8 lambda表達式的語法。你可以使用lambda寫出如下代碼:

1
2
3
(params) -> expression
(params) -> statement
(params) -> { statements }

例如,如果你的方法不對參數進行修改、重寫,只是在控制台打印點東西的話,那么可以這樣寫:

1
() -> System.out.println( "Hello Lambda Expressions" );

如果你的方法接收兩個參數,那么可以寫成如下這樣:

1
( int even, int odd) -> even + odd

順便提一句,通常都會把lambda表達式內部變量的名字起得短一些。這樣能使代碼更簡短,放在同一行。所以,在上述代碼中,變量名選用a、b或者x、y會比even、odd要好。

例2、使用Java 8 lambda表達式進行事件處理

如果你用過Swing API編程,你就會記得怎樣寫事件監聽代碼。這又是一個舊版本簡單匿名類的經典用例,但現在可以不這樣了。你可以用lambda表達式寫出更好的事件監聽代碼,如下所示:

1
2
3
4
5
6
7
8
// Java 8之前:
JButton show =  new JButton( "Show" );
show.addActionListener( new ActionListener() {
     @Override
     public void actionPerformed(ActionEvent e) {
     System.out.println( "Event handling without lambda expression is boring" );
     }
});
1
2
3
4
// Java 8方式:
show.addActionListener((e) -> {
     System.out.println( "Light, Camera, Action !! Lambda expressions Rocks" );
});

Java開發者經常使用匿名類的另一個地方是為 Collections.sort() 定制 Comparator。在Java 8中,你可以用更可讀的lambda表達式換掉丑陋的匿名類。我把這個留做練習,應該不難,可以按照我在使用lambda表達式實現 Runnable 和 ActionListener 的過程中的套路來做。

例3、使用lambda表達式對列表進行迭代

如果你使過幾年Java,你就知道針對集合類,最常見的操作就是進行迭代,並將業務邏輯應用於各個元素,例如處理訂單、交易和事件的列表。由於Java是命令式語言,Java 8之前的所有循環代碼都是順序的,即可以對其元素進行並行化處理。如果你想做並行過濾,就需要自己寫代碼,這並不是那么容易。通過引入lambda表達式和默認方法,將做什么和怎么做的問題分開了,這意味着Java集合現在知道怎樣做迭代,並可以在API層面對集合元素進行並行處理。下面的例子里,我將介紹如何在使用lambda或不使用lambda表達式的情況下迭代列表。你可以看到列表現在有了一個 forEach()  方法,它可以迭代所有對象,並將你的lambda代碼應用在其中。

1
2
3
4
5
// Java 8之前:
List features = Arrays.asList( "Lambdas" , "Default Method" , "Stream API" , "Date and Time API" );
for (String feature : features) {
     System.out.println(feature);
}
1
2
3
4
5
6
7
// Java 8之后:
List features = Arrays.asList( "Lambdas" , "Default Method" , "Stream API" , "Date and Time API" );
features.forEach(n -> System.out.println(n));
 
// 使用Java 8的方法引用更方便,方法引用由::雙冒號操作符標示,
// 看起來像C++的作用域解析運算符
features.forEach(System.out::println);

輸出:

1
2
3
4
Lambdas
Default Method
Stream API
Date and Time API

列表循環的最后一個例子展示了如何在Java 8中使用方法引用(method reference)。你可以看到C++里面的雙冒號、范圍解析操作符現在在Java 8中用來表示方法引用。

例4、使用lambda表達式和函數式接口Predicate

除了在語言層面支持函數式編程風格,Java 8也添加了一個包,叫做 java.util.function。它包含了很多類,用來支持Java的函數式編程。其中一個便是Predicate,使用 java.util.function.Predicate 函數式接口以及lambda表達式,可以向API方法添加邏輯,用更少的代碼支持更多的動態行為。下面是Java 8 Predicate 的例子,展示了過濾集合數據的多種常用方法。Predicate接口非常適用於做過濾。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public static void main(args[]){
     List languages = Arrays.asList( "Java" , "Scala" , "C++" , "Haskell" , "Lisp" );
 
     System.out.println( "Languages which starts with J :" );
     filter(languages, (str)->str.startsWith( "J" ));
 
     System.out.println( "Languages which ends with a " );
     filter(languages, (str)->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)->str.length() > 4 );
}
 
public static void filter(List names, Predicate condition) {
     for (String name: names)  {
         if (condition.test(name)) {
             System.out.println(name + " " );
         }
     }
}

輸出:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Languages which starts with J :
Java
Languages which ends with a
Java
Scala
Print all languages :
Java
Scala
C++
Haskell
Lisp
Print no language :
Print language whose length greater than 4:
Scala
Haskell
1
2
3
4
5
6
// 更好的辦法
public static void filter(List names, Predicate condition) {
     names.stream().filter((name) -> (condition.test(name))).forEach((name) -> {
         System.out.println(name + " " );
     });
}

可以看到,Stream API的過濾方法也接受一個Predicate,這意味着可以將我們定制的 filter() 方法替換成寫在里面的內聯代碼,這就是lambda表達式的魔力。另外,Predicate接口也允許進行多重條件的測試,下個例子將要講到。

例5、如何在lambda表達式中加入Predicate

上個例子說到,java.util.function.Predicate 允許將兩個或更多的 Predicate 合成一個。它提供類似於邏輯操作符AND和OR的方法,名字叫做and()、or()和xor(),用於將傳入 filter() 方法的條件合並起來。例如,要得到所有以J開始,長度為四個字母的語言,可以定義兩個獨立的 Predicate 示例分別表示每一個條件,然后用 Predicate.and() 方法將它們合並起來,如下所示:

1
2
3
4
5
6
7
// 甚至可以用and()、or()和xor()邏輯函數來合並Predicate,
// 例如要找到所有以J開始,長度為四個字母的名字,你可以合並兩個Predicate並傳入
Predicate<String> startsWithJ = (n) -> n.startsWith( "J" );
Predicate<String> fourLetterLong = (n) -> n.length() == 4 ;
names.stream()
     .filter(startsWithJ.and(fourLetterLong))
     .forEach((n) -> System.out.print( "nName, which starts with 'J' and four letter long is : " + n));

類似地,也可以使用 or() 和 xor() 方法。本例着重介紹了如下要點:可按需要將 Predicate 作為單獨條件然后將其合並起來使用。簡而言之,你可以以傳統Java命令方式使用 Predicate 接口,也可以充分利用lambda表達式達到事半功倍的效果。

例6、Java 8中使用lambda表達式的Map和Reduce示例

本例介紹最廣為人知的函數式編程概念map。它允許你將對象進行轉換。例如在本例中,我們將 costBeforeTax 列表的每個元素轉換成為稅后的值。我們將 x -> x*x lambda表達式傳到 map() 方法,后者將其應用到流中的每一個元素。然后用 forEach() 將列表元素打印出來。使用流API的收集器類,可以得到所有含稅的開銷。有 toList() 這樣的方法將 map 或任何其他操作的結果合並起來。由於收集器在流上做終端操作,因此之后便不能重用流了。你甚至可以用流API的 reduce() 方法將所有數字合成一個,下一個例子將會講到。

1
2
3
4
5
6
7
8
9
10
// 不使用lambda表達式為每個訂單加上12%的稅
List costBeforeTax = Arrays.asList( 100 , 200 , 300 , 400 , 500 );
for (Integer cost : costBeforeTax) {
     double price = cost + . 12 *cost;
     System.out.println(price);
}
 
// 使用lambda表達式
List costBeforeTax = Arrays.asList( 100 , 200 , 300 , 400 , 500 );
costBeforeTax.stream().map((cost) -> cost + . 12 *cost).forEach(System.out::println);

輸出:

1
2
3
4
5
6
7
8
9
10
112.0
224.0
336.0
448.0
560.0
112.0
224.0
336.0
448.0
560.0

例6.2、Java 8中使用lambda表達式的Map和Reduce示例

在上個例子中,可以看到map將集合類(例如列表)元素進行轉換的。還有一個 reduce() 函數可以將所有值合並成一個。Map和Reduce操作是函數式編程的核心操作,因為其功能,reduce 又被稱為折疊操作。另外,reduce 並不是一個新的操作,你有可能已經在使用它。SQL中類似 sum()、avg() 或者 count() 的聚集函數,實際上就是 reduce 操作,因為它們接收多個值並返回一個值。流API定義的 reduceh() 函數可以接受lambda表達式,並對所有值進行合並。IntStream這樣的類有類似 average()、count()、sum() 的內建方法來做 reduce 操作,也有mapToLong()、mapToDouble() 方法來做轉換。這並不會限制你,你可以用內建方法,也可以自己定義。在這個Java 8的Map Reduce示例里,我們首先對所有價格應用 12% 的VAT,然后用 reduce() 方法計算總和。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
// 為每個訂單加上12%的稅
// 老方法:
List costBeforeTax = Arrays.asList( 100 , 200 , 300 , 400 , 500 );
double total = 0 ;
for (Integer cost : costBeforeTax) {
     double price = cost + . 12 *cost;
     total = total + price;
}
System.out.println( "Total : " + total);
 
// 新方法:
List costBeforeTax = Arrays.asList( 100 , 200 , 300 , 400 , 500 );
double bill = costBeforeTax.stream().map((cost) -> cost + . 12 *cost).reduce((sum, cost) -> sum + cost).get();
System.out.println( "Total : " + bill);

輸出:

1
2
Total : 1680.0
Total : 1680.0

例7、通過過濾創建一個String列表

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

1
2
3
// 創建一個字符串列表,每個字符串長度大於2
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);

輸出:

1
Original List : [abc, , bcd, , defg, jk], filtered list : [abc, bcd, defg]

另外,關於 filter() 方法有個常見誤解。在現實生活中,做過濾的時候,通常會丟棄部分,但使用filter()方法則是獲得一個新的列表,且其每個元素符合過濾原則。

例8、對列表的每個元素應用函數

我們通常需要對列表的每個元素使用某個函數,例如逐一乘以某個數、除以某個數或者做其它操作。這些操作都很適合用 map() 方法,可以將轉換邏輯以lambda表達式的形式放在 map() 方法里,就可以對集合的各個元素進行轉換了,如下所示。

1
2
3
4
// 將字符串換成大寫並用逗號鏈接起來
List<String> G7 = Arrays.asList( "USA" , "Japan" , "France" , "Germany" , "Italy" , "U.K." , "Canada" );
String G7Countries = G7.stream().map(x -> x.toUpperCase()).collect(Collectors.joining( ", " ));
System.out.println(G7Countries);

輸出:

1
USA, JAPAN, FRANCE, GERMANY, ITALY, U.K., CANADA

例9、復制不同的值,創建一個子列表

本例展示了如何利用流的 distinct() 方法來對集合進行去重。

1
2
3
4
// 用所有不同的數字創建一個正方形列表
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);

輸出:

1
Original List : [9, 10, 3, 4, 7, 3, 4],  Square Without duplicates : [81, 100, 9, 16, 49]

例10、計算集合元素的最大值、最小值、總和以及平均值

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

1
2
3
4
5
6
7
//獲取數字的個數、最小值、最大值、總和以及平均值
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());

輸出:

1
2
3
4
Highest prime number in List : 29
Lowest prime number in List : 2
Sum of all prime numbers : 129
Average of all prime numbers : 12.9

Lambda表達式 vs 匿名類

既然lambda表達式即將正式取代Java代碼中的匿名內部類,那么有必要對二者做一個比較分析。一個關鍵的不同點就是關鍵字 this。匿名類的 this 關鍵字指向匿名類,而lambda表達式的 this 關鍵字指向包圍lambda表達式的類。另一個不同點是二者的編譯方式。Java編譯器將lambda表達式編譯成類的私有方法。使用了Java 7的 invokedynamic 字節碼指令來動態綁定這個方法。

Java 8 Lambda表達式要點

10個Java lambda表達式、流API示例

到目前為止我們看到了Java 8的10個lambda表達式,這對於新手來說是個合適的任務量,你可能需要親自運行示例程序以便掌握。試着修改要求創建自己的例子,達到快速學習的目的。我還想建議大家使用Netbeans IDE來練習lambda表達式,它對Java 8支持良好。當把代碼轉換成函數式的時候,Netbeans會及時給你提示。只需跟着Netbeans的提示,就能很容易地把匿名類轉換成lambda表達式。此外,如果你喜歡閱讀,那么記得看一下Java 8的lambdas,實用函數式編程這本書(Java 8 Lambdas, pragmatic functional programming),作者是Richard Warburton,或者也可以看看Manning的Java 8實戰(Java 8 in Action),這本書雖然還沒出版,但我猜線上有第一章的免費pdf。不過,在你開始忙其它事情之前,先回顧一下Java 8的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表達式可以換為方法引用,因為這僅是一個參數相同的簡單方法調用。

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

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

1
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表達式生成的字節碼。大致應該長這樣:

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

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

1
2
3
List<Integer> primes = Arrays.asList( new Integer[]{ 2 , 3 , 5 , 7 });
int factor = 2 ;
primes.forEach(element -> { factor++; });
1
Compile time error : "local variables referenced from a lambda expression must be final or effectively final"

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

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

輸出:

1
2
3
4
4
6
10
14

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

以上就是Java 8的lambda表達式的全部10個例子。此次修改將成為Java史上最大的一次,將深遠影響未來Java開發者使用集合框架的方式。我想規模最相似的一次修改就是Java 5的發布了,它帶來了很多優點,提升了代碼質量,例如:泛型、枚舉、自動裝箱(Autoboxing)、靜態導入、並發API和變量參數。上述特性使得Java代碼更加清晰,我想lambda表達式也將進一步改進它。我在期待着開發並行第三方庫,這可以使高性能應用變得更容易寫。

更多閱讀:http://javarevisited.blogspot.com/2014/02/10-example-of-lambda-expressions-in-java8.html#ixzz3gCMp6Vhc

原文鏈接: javarevisited 翻譯: ImportNew.com lemeilleur
譯文鏈接: http://www.importnew.com/16436.html
轉載請保留原文出處、譯者和譯文鏈接。]


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM